選択できるのは25トピックまでです。 トピックは、先頭が英数字で、英数字とダッシュ('-')を使用した35文字以内のものにしてください。

DocumentTotalViewModel.php 5.2KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  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. $conversionMessage = $this->buildConversionMessage($grandTotalInCents, $currencyCode, $defaultCurrencyCode);
  27. return [
  28. 'subtotal' => CurrencyConverter::formatCentsToMoney($subtotalInCents, $currencyCode),
  29. 'taxTotal' => CurrencyConverter::formatCentsToMoney($taxTotalInCents, $currencyCode),
  30. 'discountTotal' => CurrencyConverter::formatCentsToMoney($discountTotalInCents, $currencyCode),
  31. 'grandTotal' => CurrencyConverter::formatCentsToMoney($grandTotalInCents, $currencyCode),
  32. 'conversionMessage' => $conversionMessage,
  33. ];
  34. }
  35. private function calculateLineSubtotalInCents(array $item, string $currencyCode): int
  36. {
  37. $quantity = max((float) ($item['quantity'] ?? 0), 0);
  38. $unitPrice = max((float) ($item['unit_price'] ?? 0), 0);
  39. $subtotal = $quantity * $unitPrice;
  40. return CurrencyConverter::convertToCents($subtotal, $currencyCode);
  41. }
  42. private function calculateAdjustmentsTotalInCents($lineItems, string $key, string $currencyCode): int
  43. {
  44. return $lineItems->reduce(function ($carry, $item) use ($key, $currencyCode) {
  45. $quantity = max((float) ($item['quantity'] ?? 0), 0);
  46. $unitPrice = max((float) ($item['unit_price'] ?? 0), 0);
  47. $adjustmentIds = $item[$key] ?? [];
  48. $lineTotal = $quantity * $unitPrice;
  49. $lineTotalInCents = CurrencyConverter::convertToCents($lineTotal, $currencyCode);
  50. $adjustmentTotal = Adjustment::whereIn('id', $adjustmentIds)
  51. ->get()
  52. ->sum(function (Adjustment $adjustment) use ($lineTotalInCents) {
  53. if ($adjustment->computation->isPercentage()) {
  54. return RateCalculator::calculatePercentage($lineTotalInCents, $adjustment->getRawOriginal('rate'));
  55. } else {
  56. return $adjustment->getRawOriginal('rate');
  57. }
  58. });
  59. return $carry + $adjustmentTotal;
  60. }, 0);
  61. }
  62. private function calculateDiscountTotalInCents($lineItems, int $subtotalInCents, string $currencyCode): int
  63. {
  64. $discountMethod = DocumentDiscountMethod::parse($this->data['discount_method']) ?? DocumentDiscountMethod::PerLineItem;
  65. if ($discountMethod->isPerLineItem()) {
  66. return $this->calculateAdjustmentsTotalInCents($lineItems, $this->documentType->getDiscountKey(), $currencyCode);
  67. }
  68. $discountComputation = AdjustmentComputation::parse($this->data['discount_computation']) ?? AdjustmentComputation::Percentage;
  69. $discountRate = blank($this->data['discount_rate']) ? '0' : $this->data['discount_rate'];
  70. if ($discountComputation->isPercentage()) {
  71. $scaledDiscountRate = RateCalculator::parseLocalizedRate($discountRate);
  72. return RateCalculator::calculatePercentage($subtotalInCents, $scaledDiscountRate);
  73. }
  74. if (! CurrencyConverter::isValidAmount($discountRate)) {
  75. $discountRate = '0';
  76. }
  77. return CurrencyConverter::convertToCents($discountRate, $currencyCode);
  78. }
  79. private function buildConversionMessage(int $grandTotalInCents, string $currencyCode, string $defaultCurrencyCode): ?string
  80. {
  81. if ($currencyCode === $defaultCurrencyCode) {
  82. return null;
  83. }
  84. $rate = currency($currencyCode)->getRate();
  85. $indirectRate = 1 / $rate;
  86. $convertedTotalInCents = CurrencyConverter::convertBalance($grandTotalInCents, $currencyCode, $defaultCurrencyCode);
  87. $formattedRate = Number::format($indirectRate, maxPrecision: 10);
  88. return sprintf(
  89. 'Currency conversion: %s (%s) at an exchange rate of 1 %s = %s %s',
  90. CurrencyConverter::formatCentsToMoney($convertedTotalInCents, $defaultCurrencyCode),
  91. $defaultCurrencyCode,
  92. $currencyCode,
  93. $formattedRate,
  94. $defaultCurrencyCode
  95. );
  96. }
  97. }