123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081 |
- <?php
-
- namespace App\Utilities\Accounting;
-
- use App\Enums\Accounting\AccountType;
- use App\Models\Accounting\AccountSubtype;
- use RuntimeException;
-
- class AccountCode
- {
- public static function isValidCode($code, AccountType $type): bool
- {
- $range = self::getRangeForType($type);
-
- $mainAccountPart = explode('-', $code)[0];
-
- $numericValue = (int) $mainAccountPart;
-
- return $numericValue >= $range[0] && $numericValue <= $range[1];
- }
-
- public static function getMessage(AccountType $type): string
- {
- $range = self::getRangeForType($type);
-
- return "The account code must range from {$range[0]} to {$range[1]} for a {$type->getLabel()}.";
- }
-
- public static function getRangeForType(AccountType $type): array
- {
- return match ($type) {
- AccountType::CurrentAsset => [1000, 1499],
- AccountType::NonCurrentAsset => [1500, 1899],
- AccountType::ContraAsset => [1900, 1999],
- AccountType::CurrentLiability => [2000, 2499],
- AccountType::NonCurrentLiability => [2500, 2899],
- AccountType::ContraLiability => [2900, 2999],
- AccountType::Equity => [3000, 3899],
- AccountType::ContraEquity => [3900, 3999],
- AccountType::OperatingRevenue => [4000, 4499],
- AccountType::NonOperatingRevenue => [4500, 4899],
- AccountType::ContraRevenue => [4900, 4949],
- AccountType::UncategorizedRevenue => [4950, 4999],
- AccountType::OperatingExpense => [5000, 5499],
- AccountType::NonOperatingExpense => [5500, 5899],
- AccountType::ContraExpense => [5900, 5949],
- AccountType::UncategorizedExpense => [5950, 5999],
- };
- }
-
- public static function generate(AccountSubtype $accountSubtype): string
- {
- $subtypeName = $accountSubtype->name;
- $typeEnum = $accountSubtype->type;
- $typeValue = $typeEnum->value;
-
- $baseCode = config("chart-of-accounts.default.{$typeValue}.{$subtypeName}.base_code");
- $range = self::getRangeForType($typeEnum);
-
- $lastAccount = $accountSubtype->accounts()
- ->whereNotNull('code')
- ->orderBy('code', 'desc')
- ->first();
-
- $nextNumericValue = $lastAccount ? (int) explode('-', $lastAccount->code)[0] + 1 : (int) $baseCode;
-
- if ($nextNumericValue > $range[1]) {
- throw new RuntimeException("The account code range for a {$typeEnum->getLabel()} has been exceeded.");
- }
-
- while ($accountSubtype->accounts()->where('code', '=', (string) $nextNumericValue)->exists()) {
- $nextNumericValue++;
-
- if ($nextNumericValue > $range[1]) {
- throw new RuntimeException("The account code range for a {$typeEnum->getLabel()} has been exceeded.");
- }
- }
-
- return (string) $nextNumericValue;
- }
- }
|