| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290 | 
							- <?php
 - 
 - namespace App\Models\Accounting;
 - 
 - use App\Concerns\Blamable;
 - use App\Concerns\CompanyOwned;
 - use App\Enums\Accounting\AccountCategory;
 - use App\Enums\Accounting\AccountType;
 - use App\Enums\Accounting\PaymentMethod;
 - use App\Enums\Accounting\TransactionType;
 - use App\Filament\Company\Resources\Accounting\TransactionResource\Pages\ViewTransaction;
 - use App\Filament\Company\Resources\Purchases\BillResource\Pages\ViewBill;
 - use App\Filament\Company\Resources\Sales\InvoiceResource\Pages\ViewInvoice;
 - use App\Models\Banking\BankAccount;
 - use App\Models\Common\Client;
 - use App\Models\Common\Contact;
 - use App\Models\Common\Vendor;
 - use App\Observers\TransactionObserver;
 - use Database\Factories\Accounting\TransactionFactory;
 - use Illuminate\Database\Eloquent\Attributes\ObservedBy;
 - use Illuminate\Database\Eloquent\Builder;
 - use Illuminate\Database\Eloquent\Factories\Factory;
 - use Illuminate\Database\Eloquent\Factories\HasFactory;
 - use Illuminate\Database\Eloquent\Model;
 - use Illuminate\Database\Eloquent\Relations\BelongsTo;
 - use Illuminate\Database\Eloquent\Relations\HasMany;
 - use Illuminate\Database\Eloquent\Relations\MorphTo;
 - use Illuminate\Support\Collection;
 - 
 - #[ObservedBy(TransactionObserver::class)]
 - class Transaction extends Model
 - {
 -     use Blamable;
 -     use CompanyOwned;
 -     use HasFactory;
 - 
 -     protected $fillable = [
 -         'company_id',
 -         'account_id', // Account from Chart of Accounts (Income/Expense accounts)
 -         'bank_account_id', // Cash/Bank Account
 -         'plaid_transaction_id',
 -         'contact_id',
 -         'type', // 'deposit', 'withdrawal', 'journal'
 -         'payment_channel',
 -         'payment_method',
 -         'is_payment',
 -         'description',
 -         'notes',
 -         'reference',
 -         'amount',
 -         'pending',
 -         'reviewed',
 -         'posted_at',
 -         'created_by',
 -         'updated_by',
 -         'meta',
 -     ];
 - 
 -     protected $casts = [
 -         'type' => TransactionType::class,
 -         'payment_method' => PaymentMethod::class,
 -         'pending' => 'boolean',
 -         'reviewed' => 'boolean',
 -         'posted_at' => 'date',
 -         'meta' => 'array',
 -     ];
 - 
 -     public function account(): BelongsTo
 -     {
 -         return $this->belongsTo(Account::class, 'account_id');
 -     }
 - 
 -     public function bankAccount(): BelongsTo
 -     {
 -         return $this->belongsTo(BankAccount::class, 'bank_account_id');
 -     }
 - 
 -     public function contact(): BelongsTo
 -     {
 -         return $this->belongsTo(Contact::class, 'contact_id');
 -     }
 - 
 -     public function journalEntries(): HasMany
 -     {
 -         return $this->hasMany(JournalEntry::class, 'transaction_id');
 -     }
 - 
 -     public function transactionable(): MorphTo
 -     {
 -         return $this->morphTo();
 -     }
 - 
 -     public function payeeable(): MorphTo
 -     {
 -         return $this->morphTo();
 -     }
 - 
 -     public function isUncategorized(): bool
 -     {
 -         return $this->journalEntries->contains(fn (JournalEntry $entry) => $entry->account->isUncategorized());
 -     }
 - 
 -     public function isPayment(): bool
 -     {
 -         return $this->is_payment;
 -     }
 - 
 -     public function updateAmountIfBalanced(): void
 -     {
 -         if ($this->journalEntries->areBalanced() && $this->journalEntries->sumDebits()->formatSimple() !== $this->getAttributeValue('amount')) {
 -             $this->setAttribute('amount', $this->journalEntries->sumDebits()->getAmount());
 -             $this->save();
 -         }
 -     }
 - 
 -     public static function getBankAccountOptionsFlat(?int $excludedAccountId = null, ?int $currentBankAccountId = null, bool $excludeArchived = true): array
 -     {
 -         return BankAccount::query()
 -             ->whereHas('account', function (Builder $query) use ($excludeArchived) {
 -                 if ($excludeArchived) {
 -                     $query->where('archived', false);
 -                 }
 -             })
 -             ->with(['account' => function ($query) use ($excludeArchived) {
 -                 if ($excludeArchived) {
 -                     $query->where('archived', false);
 -                 }
 -             }])
 -             ->when($excludedAccountId, fn (Builder $query) => $query->where('account_id', '!=', $excludedAccountId))
 -             ->when($currentBankAccountId, fn (Builder $query) => $query->orWhere('id', $currentBankAccountId))
 -             ->get()
 -             ->pluck('account.name', 'id')
 -             ->toArray();
 -     }
 - 
 -     public static function getBankAccountOptions(?int $excludedAccountId = null, ?int $currentBankAccountId = null, bool $excludeArchived = true): array
 -     {
 -         return BankAccount::query()
 -             ->whereHas('account', function (Builder $query) use ($excludeArchived) {
 -                 if ($excludeArchived) {
 -                     $query->where('archived', false);
 -                 }
 -             })
 -             ->with(['account' => function ($query) use ($excludeArchived) {
 -                 if ($excludeArchived) {
 -                     $query->where('archived', false);
 -                 }
 -             }, 'account.subtype' => function ($query) {
 -                 $query->select(['id', 'name']);
 -             }])
 -             ->when($excludedAccountId, fn (Builder $query) => $query->where('account_id', '!=', $excludedAccountId))
 -             ->when($currentBankAccountId, fn (Builder $query) => $query->orWhere('id', $currentBankAccountId))
 -             ->get()
 -             ->groupBy('account.subtype.name')
 -             ->map(fn (Collection $bankAccounts, string $subtype) => $bankAccounts->pluck('account.name', 'id'))
 -             ->toArray();
 -     }
 - 
 -     public static function getBankAccountAccountOptions(?int $excludedBankAccountId = null, ?int $currentAccountId = null): array
 -     {
 -         return Account::query()
 -             ->whereHas('bankAccount', function (Builder $query) use ($excludedBankAccountId) {
 -                 // Exclude the specific bank account if provided
 -                 if ($excludedBankAccountId) {
 -                     $query->whereNot('id', $excludedBankAccountId);
 -                 }
 -             })
 -             ->where(function (Builder $query) use ($currentAccountId) {
 -                 $query->where('archived', false)
 -                     ->orWhere('id', $currentAccountId);
 -             })
 -             ->get()
 -             ->groupBy(fn (Account $account) => $account->category->getPluralLabel())
 -             ->map(fn (Collection $accounts, string $category) => $accounts->pluck('name', 'id'))
 -             ->toArray();
 -     }
 - 
 -     public static function getChartAccountOptions(): array
 -     {
 -         return Account::query()
 -             ->select(['id', 'name', 'category'])
 -             ->get()
 -             ->groupBy(fn (Account $account) => $account->category->getPluralLabel())
 -             ->map(fn (Collection $accounts, string $category) => $accounts->pluck('name', 'id'))
 -             ->toArray();
 -     }
 - 
 -     public static function getTransactionAccountOptions(
 -         TransactionType $type,
 -         ?int $currentAccountId = null
 -     ): array {
 -         $associatedAccountTypes = match ($type) {
 -             TransactionType::Deposit => [
 -                 AccountType::OperatingRevenue,     // Sales, service income
 -                 AccountType::NonOperatingRevenue,  // Interest, dividends received
 -                 AccountType::CurrentLiability,     // Loans received
 -                 AccountType::NonCurrentLiability,  // Long-term financing
 -                 AccountType::Equity,               // Owner contributions
 -                 AccountType::ContraExpense,        // Refunds of expenses
 -                 AccountType::UncategorizedRevenue,
 -             ],
 -             TransactionType::Withdrawal => [
 -                 AccountType::OperatingExpense,     // Regular business expenses
 -                 AccountType::NonOperatingExpense,  // Interest paid, etc.
 -                 AccountType::CurrentLiability,     // Loan payments
 -                 AccountType::NonCurrentLiability,  // Long-term debt payments
 -                 AccountType::Equity,               // Owner withdrawals
 -                 AccountType::ContraRevenue,        // Customer refunds, discounts
 -                 AccountType::UncategorizedExpense,
 -             ],
 -             default => null,
 -         };
 - 
 -         return Account::query()
 -             ->doesntHave('adjustment')
 -             ->doesntHave('bankAccount')
 -             ->when($associatedAccountTypes, fn (Builder $query) => $query->whereIn('type', $associatedAccountTypes))
 -             ->where(function (Builder $query) use ($currentAccountId) {
 -                 $query->where('archived', false)
 -                     ->orWhere('id', $currentAccountId);
 -             })
 -             ->get()
 -             ->groupBy(fn (Account $account) => $account->category->getPluralLabel())
 -             ->map(fn (Collection $accounts, string $category) => $accounts->pluck('name', 'id'))
 -             ->toArray();
 -     }
 - 
 -     public static function getJournalAccountOptions(
 -         ?int $currentAccountId = null
 -     ): array {
 -         return Account::query()
 -             ->where(function (Builder $query) use ($currentAccountId) {
 -                 $query->where('archived', false)
 -                     ->orWhere('id', $currentAccountId);
 -             })
 -             ->get()
 -             ->groupBy(fn (Account $account) => $account->category->getPluralLabel())
 -             ->map(fn (Collection $accounts, string $category) => $accounts->pluck('name', 'id'))
 -             ->toArray();
 -     }
 - 
 -     public static function getUncategorizedAccountByType(TransactionType $type): ?Account
 -     {
 -         [$category, $accountName] = match ($type) {
 -             TransactionType::Deposit => [AccountCategory::Revenue, 'Uncategorized Income'],
 -             TransactionType::Withdrawal => [AccountCategory::Expense, 'Uncategorized Expense'],
 -             default => [null, null],
 -         };
 - 
 -         return Account::where('category', $category)
 -             ->where('name', $accountName)
 -             ->first();
 -     }
 - 
 -     public static function getPayeeOptions(): array
 -     {
 -         $clients = Client::query()
 -             ->orderBy('name')
 -             ->pluck('name', 'id')
 -             ->toArray();
 - 
 -         $vendors = Vendor::query()
 -             ->orderBy('name')
 -             ->pluck('name', 'id')
 -             ->mapWithKeys(fn ($name, $id) => [-$id => $name])
 -             ->toArray();
 - 
 -         return [
 -             'Clients' => $clients,
 -             'Vendors' => $vendors,
 -         ];
 -     }
 - 
 -     public function getReportTableUrl(): string
 -     {
 -         if ($this->transactionable_type && ! $this->is_payment) {
 -             return match ($this->transactionable_type) {
 -                 Bill::class => ViewBill::getUrl(['record' => $this->transactionable_id]),
 -                 default => ViewInvoice::getUrl(['record' => $this->transactionable_id]),
 -             };
 -         }
 - 
 -         return ViewTransaction::getUrl(['record' => $this->id]);
 -     }
 - 
 -     protected static function newFactory(): Factory
 -     {
 -         return TransactionFactory::new();
 -     }
 - }
 
 
  |