priceCurrency = $priceCurrency; } /** * Collects credit memo shipping totals. * * @param \Magento\Sales\Model\Order\Creditmemo $creditmemo * @return $this * @throws \Magento\Framework\Exception\LocalizedException */ public function collect(\Magento\Sales\Model\Order\Creditmemo $creditmemo) { $order = $creditmemo->getOrder(); // amounts without tax $orderShippingAmount = $order->getShippingAmount(); $orderBaseShippingAmount = $order->getBaseShippingAmount(); $allowedAmount = $orderShippingAmount - $order->getShippingRefunded(); $baseAllowedAmount = $orderBaseShippingAmount - $order->getBaseShippingRefunded(); // amounts including tax $orderShippingInclTax = $order->getShippingInclTax(); $orderBaseShippingInclTax = $order->getBaseShippingInclTax(); $allowedTaxAmount = $order->getShippingTaxAmount() - $order->getShippingTaxRefunded(); $allowedAmountInclTax = $allowedAmount + $allowedTaxAmount; $baseAllowedAmountInclTax = $orderBaseShippingInclTax - $order->getBaseShippingRefunded() - $order->getBaseShippingTaxRefunded(); // Check if the desired shipping amount to refund was specified (from invoice or another source). if ($creditmemo->hasBaseShippingAmount()) { // For the conditional logic, we will either use amounts that always include tax -OR- never include tax. // The logic uses the 'base' currency to be consistent with what the user (admin) provided as input. $useAmountsWithTax = $this->isSuppliedShippingAmountInclTax($order); // Since the user (admin) supplied 'desiredAmount' it already has tax -OR- does not include tax $desiredAmount = $this->priceCurrency->round($creditmemo->getBaseShippingAmount()); $maxAllowedAmount = ($useAmountsWithTax ? $baseAllowedAmountInclTax : $baseAllowedAmount); $originalTotalAmount = ($useAmountsWithTax ? $orderBaseShippingInclTax : $orderBaseShippingAmount); // Note: ($x < $y + 0.0001) means ($x <= $y) for floats if ($desiredAmount < $this->priceCurrency->round($maxAllowedAmount) + 0.0001) { // since the admin is returning less than the allowed amount, compute the ratio being returned $ratio = 0; if ($originalTotalAmount > 0) { $ratio = $desiredAmount / $originalTotalAmount; } // capture amounts without tax // Note: ($x > $y - 0.0001) means ($x >= $y) for floats if ($desiredAmount > $maxAllowedAmount - 0.0001) { $shippingAmount = $allowedAmount; $baseShippingAmount = $baseAllowedAmount; } else { $shippingAmount = $this->priceCurrency->round($orderShippingAmount * $ratio); $baseShippingAmount = $this->priceCurrency->round($orderBaseShippingAmount * $ratio); } $shippingInclTax = $this->priceCurrency->round($orderShippingInclTax * $ratio); $baseShippingInclTax = $this->priceCurrency->round($orderBaseShippingInclTax * $ratio); } else { $maxAllowedAmount = $order->getBaseCurrency()->format($maxAllowedAmount, null, false); throw new \Magento\Framework\Exception\LocalizedException( __('Maximum shipping amount allowed to refund is: %1', $maxAllowedAmount) ); } } else { $shippingAmount = $allowedAmount; $baseShippingAmount = $baseAllowedAmount; $shippingInclTax = $this->priceCurrency->round($allowedAmountInclTax); $baseShippingInclTax = $this->priceCurrency->round($baseAllowedAmountInclTax); } $creditmemo->setShippingAmount($shippingAmount); $creditmemo->setBaseShippingAmount($baseShippingAmount); $creditmemo->setShippingInclTax($shippingInclTax); $creditmemo->setBaseShippingInclTax($baseShippingInclTax); $creditmemo->setGrandTotal($creditmemo->getGrandTotal() + $shippingAmount); $creditmemo->setBaseGrandTotal($creditmemo->getBaseGrandTotal() + $baseShippingAmount); return $this; } /** * Returns whether the user specified a shipping amount that already includes tax * * @param \Magento\Sales\Model\Order $order * @return bool */ private function isSuppliedShippingAmountInclTax($order) { // returns true if we are only displaying shipping including tax, otherwise returns false return $this->getTaxConfig()->displaySalesShippingInclTax($order->getStoreId()); } /** * Get the Tax Config. * * @return \Magento\Tax\Model\Config * * @deprecated 100.1.0 */ private function getTaxConfig() { if ($this->taxConfig === null) { $this->taxConfig = \Magento\Framework\App\ObjectManager::getInstance()->get( \Magento\Tax\Model\Config::class ); } return $this->taxConfig; } }