QuantityValidator.php 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251
  1. <?php
  2. /**
  3. * Copyright © Magento, Inc. All rights reserved.
  4. * See COPYING.txt for license details.
  5. */
  6. namespace Magento\Sales\Model\Order\Creditmemo\Validation;
  7. use Magento\Sales\Api\Data\CreditmemoInterface;
  8. use Magento\Sales\Api\Data\OrderInterface;
  9. use Magento\Sales\Api\Data\OrderItemInterface;
  10. use Magento\Sales\Api\InvoiceRepositoryInterface;
  11. use Magento\Sales\Api\OrderRepositoryInterface;
  12. use Magento\Sales\Model\Order;
  13. use Magento\Sales\Model\Order\Creditmemo;
  14. use Magento\Sales\Model\Order\Item;
  15. use Magento\Sales\Model\ValidatorInterface;
  16. /**
  17. * Class QuantityValidator
  18. */
  19. class QuantityValidator implements ValidatorInterface
  20. {
  21. /**
  22. * @var OrderRepositoryInterface
  23. */
  24. private $orderRepository;
  25. /**
  26. * @var InvoiceRepositoryInterface
  27. */
  28. private $invoiceRepository;
  29. /**
  30. * @var \Magento\Framework\Pricing\PriceCurrencyInterface
  31. */
  32. private $priceCurrency;
  33. /**
  34. * InvoiceValidator constructor.
  35. *
  36. * @param OrderRepositoryInterface $orderRepository
  37. * @param InvoiceRepositoryInterface $invoiceRepository
  38. * @param \Magento\Framework\Pricing\PriceCurrencyInterface $priceCurrency
  39. */
  40. public function __construct(
  41. OrderRepositoryInterface $orderRepository,
  42. InvoiceRepositoryInterface $invoiceRepository,
  43. \Magento\Framework\Pricing\PriceCurrencyInterface $priceCurrency
  44. ) {
  45. $this->orderRepository = $orderRepository;
  46. $this->invoiceRepository = $invoiceRepository;
  47. $this->priceCurrency = $priceCurrency;
  48. }
  49. /**
  50. * @inheritdoc
  51. */
  52. public function validate($entity)
  53. {
  54. /**
  55. * @var $entity CreditmemoInterface
  56. */
  57. if ($entity->getOrderId() === null) {
  58. return [__('Order Id is required for creditmemo document')];
  59. }
  60. $messages = [];
  61. $order = $this->orderRepository->get($entity->getOrderId());
  62. $orderItemsById = $this->getOrderItems($order);
  63. $invoiceQtysRefundLimits = $this->getInvoiceQtysRefundLimits($entity, $order);
  64. $totalQuantity = 0;
  65. foreach ($entity->getItems() as $item) {
  66. if (!isset($orderItemsById[$item->getOrderItemId()])) {
  67. $messages[] = __(
  68. 'The creditmemo contains product SKU "%1" that is not part of the original order.',
  69. $item->getSku()
  70. );
  71. continue;
  72. }
  73. $orderItem = $orderItemsById[$item->getOrderItemId()];
  74. if (!$this->canRefundItem($orderItem, $item->getQty(), $invoiceQtysRefundLimits) ||
  75. !$this->isQtyAvailable($orderItem, $item->getQty())
  76. ) {
  77. $messages[] =__(
  78. 'The quantity to creditmemo must not be greater than the unrefunded quantity'
  79. . ' for product SKU "%1".',
  80. $orderItem->getSku()
  81. );
  82. } else {
  83. $totalQuantity += $item->getQty();
  84. }
  85. }
  86. if ($entity->getGrandTotal() <= 0) {
  87. $messages[] = __('The credit memo\'s total must be positive.');
  88. } elseif ($totalQuantity <= 0 && !$this->canRefundShipping($order)) {
  89. $messages[] = __('You can\'t create a creditmemo without products.');
  90. }
  91. return $messages;
  92. }
  93. /**
  94. * We can have problem with float in php (on some server $a=762.73;$b=762.73; $a-$b!=0)
  95. * for this we have additional diapason for 0
  96. * TotalPaid - contains amount, that were not rounded.
  97. *
  98. * @param OrderInterface $order
  99. * @return bool
  100. */
  101. private function canRefundShipping(OrderInterface $order)
  102. {
  103. return !abs($this->priceCurrency->round($order->getShippingAmount()) - $order->getShippingRefunded()) < .0001;
  104. }
  105. /**
  106. * @param CreditmemoInterface $creditmemo
  107. * @param OrderInterface $order
  108. * @return array
  109. */
  110. private function getInvoiceQtysRefundLimits(CreditmemoInterface $creditmemo, OrderInterface $order)
  111. {
  112. $invoiceQtysRefundLimits = [];
  113. if ($creditmemo->getInvoiceId() !== null) {
  114. $invoiceQtysRefunded = [];
  115. $invoice = $this->invoiceRepository->get($creditmemo->getInvoiceId());
  116. foreach ($order->getCreditmemosCollection() as $createdCreditmemo) {
  117. if ($createdCreditmemo->getState() != Creditmemo::STATE_CANCELED &&
  118. $createdCreditmemo->getInvoiceId() == $invoice->getId()
  119. ) {
  120. foreach ($createdCreditmemo->getAllItems() as $createdCreditmemoItem) {
  121. $orderItemId = $createdCreditmemoItem->getOrderItem()->getId();
  122. if (isset($invoiceQtysRefunded[$orderItemId])) {
  123. $invoiceQtysRefunded[$orderItemId] += $createdCreditmemoItem->getQty();
  124. } else {
  125. $invoiceQtysRefunded[$orderItemId] = $createdCreditmemoItem->getQty();
  126. }
  127. }
  128. }
  129. }
  130. foreach ($invoice->getItems() as $invoiceItem) {
  131. $invoiceQtyCanBeRefunded = $invoiceItem->getQty();
  132. $orderItemId = $invoiceItem->getOrderItem()->getId();
  133. if (isset($invoiceQtysRefunded[$orderItemId])) {
  134. $invoiceQtyCanBeRefunded = $invoiceQtyCanBeRefunded - $invoiceQtysRefunded[$orderItemId];
  135. }
  136. $invoiceQtysRefundLimits[$orderItemId] = $invoiceQtyCanBeRefunded;
  137. }
  138. }
  139. return $invoiceQtysRefundLimits;
  140. }
  141. /**
  142. * @param OrderInterface $order
  143. * @return OrderItemInterface[]
  144. */
  145. private function getOrderItems(OrderInterface $order)
  146. {
  147. $orderItemsById = [];
  148. foreach ($order->getItems() as $item) {
  149. $orderItemsById[$item->getItemId()] = $item;
  150. }
  151. return $orderItemsById;
  152. }
  153. /**
  154. * @param Item $orderItem
  155. * @param int $qty
  156. * @return bool
  157. */
  158. private function isQtyAvailable(Item $orderItem, $qty)
  159. {
  160. return $qty <= $orderItem->getQtyToRefund() || $orderItem->isDummy();
  161. }
  162. /**
  163. * Check if order item can be refunded
  164. *
  165. * @param \Magento\Sales\Model\Order\Item $item
  166. * @param double $qty
  167. * @param array $invoiceQtysRefundLimits
  168. * @return bool
  169. */
  170. private function canRefundItem(\Magento\Sales\Model\Order\Item $item, $qty, array $invoiceQtysRefundLimits)
  171. {
  172. if ($item->isDummy()) {
  173. return $this->canRefundDummyItem($item, $qty, $invoiceQtysRefundLimits);
  174. }
  175. return $this->canRefundNoDummyItem($item, $invoiceQtysRefundLimits);
  176. }
  177. /**
  178. * Check if no dummy order item can be refunded
  179. *
  180. * @param \Magento\Sales\Model\Order\Item $item
  181. * @param array $invoiceQtysRefundLimits
  182. * @return bool
  183. */
  184. private function canRefundNoDummyItem(\Magento\Sales\Model\Order\Item $item, array $invoiceQtysRefundLimits = [])
  185. {
  186. if ($item->getQtyToRefund() < 0) {
  187. return false;
  188. }
  189. if (isset($invoiceQtysRefundLimits[$item->getId()])) {
  190. return $invoiceQtysRefundLimits[$item->getId()] > 0;
  191. }
  192. return true;
  193. }
  194. /**
  195. * @param Item $item
  196. * @param int $qty
  197. * @param array $invoiceQtysRefundLimits
  198. * @return bool
  199. */
  200. private function canRefundDummyItem(\Magento\Sales\Model\Order\Item $item, $qty, array $invoiceQtysRefundLimits)
  201. {
  202. if ($item->getHasChildren()) {
  203. foreach ($item->getChildrenItems() as $child) {
  204. if ($this->canRefundRequestedQty($child, $qty, $invoiceQtysRefundLimits)) {
  205. return true;
  206. }
  207. }
  208. } elseif ($item->getParentItem()) {
  209. return $this->canRefundRequestedQty($item->getParentItem(), $qty, $invoiceQtysRefundLimits);
  210. }
  211. return false;
  212. }
  213. /**
  214. * @param Item $item
  215. * @param int $qty
  216. * @param array $invoiceQtysRefundLimits
  217. * @return bool
  218. */
  219. private function canRefundRequestedQty(
  220. \Magento\Sales\Model\Order\Item $item,
  221. $qty,
  222. array $invoiceQtysRefundLimits
  223. ) {
  224. return $qty === null ? $this->canRefundNoDummyItem($item, $invoiceQtysRefundLimits) : $qty > 0;
  225. }
  226. }