Você não pode selecionar mais de 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.

DocumentTotalViewModel.php 6.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. <?php
  2. namespace App\View\Models;
  3. use App\Enums\Accounting\AdjustmentComputation;
  4. use App\Enums\Accounting\DocumentDiscountMethod;
  5. use App\Enums\Accounting\DocumentType;
  6. use App\Models\Accounting\Adjustment;
  7. use App\Utilities\Currency\CurrencyAccessor;
  8. use App\Utilities\Currency\CurrencyConverter;
  9. use App\Utilities\RateCalculator;
  10. use Illuminate\Support\Number;
  11. class DocumentTotalViewModel
  12. {
  13. public function __construct(
  14. public ?array $data,
  15. public DocumentType $documentType = DocumentType::Invoice,
  16. ) {}
  17. public function buildViewData(): array
  18. {
  19. $currencyCode = $this->data['currency_code'] ?? CurrencyAccessor::getDefaultCurrency();
  20. $defaultCurrencyCode = CurrencyAccessor::getDefaultCurrency();
  21. $lineItems = collect($this->data['lineItems'] ?? []);
  22. $subtotalInCents = $lineItems->sum(fn ($item) => $this->calculateLineSubtotalInCents($item, $currencyCode));
  23. $taxTotalInCents = $this->calculateAdjustmentsTotalInCents($lineItems, $this->documentType->getTaxKey(), $currencyCode);
  24. $discountTotalInCents = $this->calculateDiscountTotalInCents($lineItems, $subtotalInCents, $currencyCode);
  25. $grandTotalInCents = $subtotalInCents + ($taxTotalInCents - $discountTotalInCents);
  26. $amountDueInCents = $this->calculateAmountDueInCents($grandTotalInCents, $currencyCode);
  27. $conversionMessage = $this->buildConversionMessage($grandTotalInCents, $currencyCode, $defaultCurrencyCode);
  28. $discountMethod = DocumentDiscountMethod::parse($this->data['discount_method']);
  29. $isPerDocumentDiscount = $discountMethod->isPerDocument();
  30. $taxTotal = $taxTotalInCents > 0
  31. ? CurrencyConverter::formatCentsToMoney($taxTotalInCents, $currencyCode)
  32. : null;
  33. $discountTotal = ($isPerDocumentDiscount || $discountTotalInCents > 0)
  34. ? CurrencyConverter::formatCentsToMoney($discountTotalInCents, $currencyCode)
  35. : null;
  36. $subtotal = ($taxTotal || $discountTotal)
  37. ? CurrencyConverter::formatCentsToMoney($subtotalInCents, $currencyCode)
  38. : null;
  39. $grandTotal = CurrencyConverter::formatCentsToMoney($grandTotalInCents, $currencyCode);
  40. $amountDue = $this->documentType !== DocumentType::Estimate
  41. ? CurrencyConverter::formatCentsToMoney($amountDueInCents, $currencyCode)
  42. : null;
  43. return [
  44. 'subtotal' => $subtotal,
  45. 'taxTotal' => $taxTotal,
  46. 'discountTotal' => $discountTotal,
  47. 'grandTotal' => $grandTotal,
  48. 'amountDue' => $amountDue,
  49. 'currencyCode' => $currencyCode,
  50. 'conversionMessage' => $conversionMessage,
  51. 'isPerDocumentDiscount' => $isPerDocumentDiscount,
  52. ];
  53. }
  54. private function calculateLineSubtotalInCents(array $item, string $currencyCode): int
  55. {
  56. $quantity = max((float) ($item['quantity'] ?? 0), 0);
  57. $unitPrice = CurrencyConverter::isValidAmount($item['unit_price'], 'USD')
  58. ? CurrencyConverter::convertToFloat($item['unit_price'], 'USD')
  59. : 0;
  60. $subtotal = $quantity * $unitPrice;
  61. return CurrencyConverter::convertToCents($subtotal, 'USD');
  62. }
  63. private function calculateAdjustmentsTotalInCents($lineItems, string $key, string $currencyCode): int
  64. {
  65. return $lineItems->reduce(function ($carry, $item) use ($key) {
  66. $quantity = max((float) ($item['quantity'] ?? 0), 0);
  67. $unitPrice = CurrencyConverter::isValidAmount($item['unit_price'], 'USD')
  68. ? CurrencyConverter::convertToFloat($item['unit_price'], 'USD')
  69. : 0;
  70. $adjustmentIds = $item[$key] ?? [];
  71. $lineTotal = $quantity * $unitPrice;
  72. $lineTotalInCents = CurrencyConverter::convertToCents($lineTotal, 'USD');
  73. $adjustmentTotal = Adjustment::whereIn('id', $adjustmentIds)
  74. ->get()
  75. ->sum(function (Adjustment $adjustment) use ($lineTotalInCents) {
  76. if ($adjustment->computation->isPercentage()) {
  77. return RateCalculator::calculatePercentage($lineTotalInCents, $adjustment->getRawOriginal('rate'));
  78. } else {
  79. return $adjustment->getRawOriginal('rate');
  80. }
  81. });
  82. return $carry + $adjustmentTotal;
  83. }, 0);
  84. }
  85. private function calculateDiscountTotalInCents($lineItems, int $subtotalInCents, string $currencyCode): int
  86. {
  87. $discountMethod = DocumentDiscountMethod::parse($this->data['discount_method']) ?? DocumentDiscountMethod::PerLineItem;
  88. if ($discountMethod->isPerLineItem()) {
  89. return $this->calculateAdjustmentsTotalInCents($lineItems, $this->documentType->getDiscountKey(), $currencyCode);
  90. }
  91. $discountComputation = AdjustmentComputation::parse($this->data['discount_computation']) ?? AdjustmentComputation::Percentage;
  92. $discountRate = blank($this->data['discount_rate']) ? '0' : $this->data['discount_rate'];
  93. if ($discountComputation->isPercentage()) {
  94. $scaledDiscountRate = RateCalculator::parseLocalizedRate($discountRate);
  95. return RateCalculator::calculatePercentage($subtotalInCents, $scaledDiscountRate);
  96. }
  97. if (! CurrencyConverter::isValidAmount($discountRate)) {
  98. $discountRate = '0';
  99. }
  100. return CurrencyConverter::convertToCents($discountRate, $currencyCode);
  101. }
  102. private function buildConversionMessage(int $grandTotalInCents, string $currencyCode, string $defaultCurrencyCode): ?string
  103. {
  104. if ($currencyCode === $defaultCurrencyCode) {
  105. return null;
  106. }
  107. $rate = currency($currencyCode)->getRate();
  108. $indirectRate = 1 / $rate;
  109. $convertedTotalInCents = CurrencyConverter::convertBalance($grandTotalInCents, $currencyCode, $defaultCurrencyCode);
  110. $formattedRate = Number::format($indirectRate, maxPrecision: 10);
  111. return sprintf(
  112. 'Currency conversion: %s (%s) at an exchange rate of 1 %s = %s %s',
  113. CurrencyConverter::formatCentsToMoney($convertedTotalInCents, $defaultCurrencyCode),
  114. $defaultCurrencyCode,
  115. $currencyCode,
  116. $formattedRate,
  117. $defaultCurrencyCode
  118. );
  119. }
  120. private function calculateAmountDueInCents(int $grandTotalInCents, string $currencyCode): int
  121. {
  122. $amountPaidInCents = $this->data['amount_paid'] ?? 0;
  123. return $grandTotalInCents - $amountPaidInCents;
  124. }
  125. }