|
- <?php
- /**
- Copyright Guillaume Bourgeois (2018)
-
- contact@souke.fr
-
- Ce logiciel est un programme informatique servant à aider les producteurs
- à distribuer leur production en circuits courts.
-
- Ce logiciel est régi par la licence CeCILL soumise au droit français et
- respectant les principes de diffusion des logiciels libres. Vous pouvez
- utiliser, modifier et/ou redistribuer ce programme sous les conditions
- de la licence CeCILL telle que diffusée par le CEA, le CNRS et l'INRIA
- sur le site "http://www.cecill.info".
-
- En contrepartie de l'accessibilité au code source et des droits de copie,
- de modification et de redistribution accordés par cette licence, il n'est
- offert aux utilisateurs qu'une garantie limitée. Pour les mêmes raisons,
- seule une responsabilité restreinte pèse sur l'auteur du programme, le
- titulaire des droits patrimoniaux et les concédants successifs.
-
- A cet égard l'attention de l'utilisateur est attirée sur les risques
- associés au chargement, à l'utilisation, à la modification et/ou au
- développement et à la reproduction du logiciel par l'utilisateur étant
- donné sa spécificité de logiciel libre, qui peut le rendre complexe à
- manipuler et qui le réserve donc à des développeurs et des professionnels
- avertis possédant des connaissances informatiques approfondies. Les
- utilisateurs sont donc invités à charger et tester l'adéquation du
- logiciel à leurs besoins dans des conditions permettant d'assurer la
- sécurité de leurs systèmes et ou de leurs données et, plus généralement,
- à l'utiliser et l'exploiter dans les mêmes conditions de sécurité.
-
- Le fait que vous puissiez accéder à cet en-tête signifie que vous avez
- pris connaissance de la licence CeCILL, et que vous en avez accepté les
- termes.
- */
-
- namespace domain\Order\Order;
-
- use domain\_\AbstractResolver;
- use domain\Distribution\Distribution\Distribution;
- use domain\Distribution\ProductDistribution\ProductDistributionRepository;
- use domain\Feature\Feature\Feature;
- use domain\Feature\Feature\FeatureChecker;
- use domain\Product\Accessory\Accessory;
- use domain\Product\Product\Product;
- use domain\Product\Product\ProductSolver;
-
- class OrderResolver extends AbstractResolver
- {
- protected ProductSolver $productSolver;
- protected OrderSolver $orderSolver;
- protected OrderRepository $orderRepository;
- protected ProductDistributionRepository $productDistributionRepository;
- protected FeatureChecker $featureChecker;
- protected OrderBuilder $orderBuilder;
-
- protected array $ordersArrayCachedByDistribution = [];
-
- public function loadDependencies(): void
- {
- $this->productSolver = $this->loadService(ProductSolver::class);
- $this->orderSolver = $this->loadService(OrderSolver::class);
- $this->orderRepository = $this->loadService(OrderRepository::class);
- $this->productDistributionRepository = $this->loadService(ProductDistributionRepository::class);
- $this->featureChecker = $this->loadService(FeatureChecker::class);
- $this->orderBuilder = $this->loadService(OrderBuilder::class);
- }
-
- public function getProductQuantityByDistribution(
- Product $product,
- Distribution $distribution,
- bool $inNumberOfPieces = false,
- Order $orderCurrent = null
- ): float
- {
- if(!isset($this->ordersArrayCachedByDistribution[$distribution->id])) {
- $this->ordersArrayCachedByDistribution[$distribution->id] = $this->orderRepository->findOrdersByDistribution($distribution);
- }
- $ordersArray = $this->ordersArrayCachedByDistribution[$distribution->id];
- if($orderCurrent) {
- $ordersArray = $this->orderBuilder->addOrUpdateOrIgnoreOrderFromArray($orderCurrent, $ordersArray);
- }
-
- $productQuantity = $this->orderSolver->getProductQuantity($product, $ordersArray);
-
- if($inNumberOfPieces) {
- $productQuantity = $this->productSolver->getNumberOfPieces($productQuantity, $product);
- }
-
- return $productQuantity;
- }
-
- public function getSmallestQuantityAccessory(Product $product): ?int
- {
- $smallestQuantity = null;
-
- foreach ($product->getProductAccessories() as $productAccessory) {
- $quantityAccessory = $productAccessory->getAccessory()->getQuantity();
- $smallestQuantity = is_null($smallestQuantity) ? $quantityAccessory
- : min($smallestQuantity, $quantityAccessory);
- }
-
- return $smallestQuantity;
- }
-
- public function getProductQuantityMax(Product $product, Distribution $distribution): ?float
- {
- $productDistribution = $this->productDistributionRepository->findOneProductDistribution($distribution, $product);
- if(!$productDistribution) {
- return null;
- }
-
- $quantityMax = $productDistribution->quantity_max;
-
- if($this->featureChecker->isEnabled(Feature::ALIAS_PRODUCT_ACCESSORY)) {
- $smallestQuantityAccessory = $this->getSmallestQuantityAccessory($product);
- if (!is_null($smallestQuantityAccessory)) {
- $smallestQuantityAccessory = $this->productSolver->getWeightOrNumberOfPieces($smallestQuantityAccessory, $product);
- $quantityMax = is_null($quantityMax) ? $smallestQuantityAccessory
- : min($quantityMax, $smallestQuantityAccessory);
- }
- }
-
- return $quantityMax;
- }
-
- public function getProductQuantityMaxOrderable(Product $product, Distribution $distribution, Order $orderCurrent = null): ?float
- {
- $productQuantity = $this->orderSolver->getProductQuantity($product, $orderCurrent ? [$orderCurrent] : []);
- $productQuantityRemaining = $this->getProductQuantityRemaining($product, $distribution, $orderCurrent);
-
- if(is_null($productQuantityRemaining)) {
- return null;
- }
-
- return $productQuantity + $productQuantityRemaining;
- }
-
- public function getQuantityOfAccessoryAvailableInDistribution(
- Accessory $accessory,
- Distribution $distribution,
- Order $orderCurrent = null
- ): int
- {
- $quantityOfAccessoryUsed = 0;
-
- foreach($accessory->getProductAccessories() as $productAccessory) {
- $quantityOfAccessoryUsed += $this->getProductQuantityByDistribution(
- $productAccessory->getProduct(),
- $distribution,
- true,
- $orderCurrent
- );
- }
-
- return $accessory->getQuantity() - $quantityOfAccessoryUsed;
- }
-
- public function getSmallestQuantityAccessoryAvailable(
- Product $product,
- Distribution $distribution,
- Order $orderCurrent = null
- ): ?int
- {
- $smallestQuantity = null;
-
- if($this->featureChecker->isEnabled(Feature::ALIAS_PRODUCT_ACCESSORY)) {
- foreach ($product->getProductAccessories() as $productAccessory) {
- $quantityAccessoryAvailableInDistribution = $this->getQuantityOfAccessoryAvailableInDistribution(
- $productAccessory->getAccessory(),
- $distribution,
- $orderCurrent
- );
- $smallestQuantity = is_null($smallestQuantity) ? $quantityAccessoryAvailableInDistribution
- : min($smallestQuantity, $quantityAccessoryAvailableInDistribution);
- }
- }
-
- return $smallestQuantity;
- }
-
- public function getProductQuantityRemaining(
- Product $product,
- Distribution $distribution,
- Order $orderCurrent = null
- ): ?float
- {
- $productDistribution = $this->productDistributionRepository->findOneProductDistribution($distribution, $product);
- if(!$productDistribution) {
- return null;
- }
-
- $quantityOrder = $this->getProductQuantityByDistribution(
- $product,
- $distribution,
- false,
- $orderCurrent
- );
- $quantityRemaining = is_null($productDistribution->quantity_max) ? null
- : ($productDistribution->quantity_max - $quantityOrder);
-
- // Limitation par nombre d'accessoires disponibles
- if($this->featureChecker->isEnabled(Feature::ALIAS_PRODUCT_ACCESSORY)) {
- $smallestQuantityAccessoryAvailable = $this->getSmallestQuantityAccessoryAvailable(
- $product,
- $distribution,
- $orderCurrent
- );
- if (!is_null($smallestQuantityAccessoryAvailable)) {
- $smallestQuantityAccessoryAvailable = $this->productSolver->getWeightOrNumberOfPieces($smallestQuantityAccessoryAvailable, $product);
- $quantityRemaining = is_null($quantityRemaining) ? $smallestQuantityAccessoryAvailable
- : min($quantityRemaining, $smallestQuantityAccessoryAvailable);
- }
- }
-
- return $quantityRemaining;
- }
- }
|