You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

598 lines
20KB

  1. <?php
  2. namespace Lc\CaracoleBundle\Repository\Order;
  3. use Doctrine\ORM\EntityManagerInterface;
  4. use Lc\CaracoleBundle\Builder\File\DocumentBuilder;
  5. use Lc\CaracoleBundle\Model\Order\OrderShopInterface;
  6. use Lc\CaracoleBundle\Model\Order\OrderStatusModel;
  7. use Lc\CaracoleBundle\Model\Reduction\ReductionCartInterface;
  8. use Lc\CaracoleBundle\Model\Reduction\ReductionCreditInterface;
  9. use Lc\CaracoleBundle\Model\Reduction\ReductionCreditModel;
  10. use Lc\CaracoleBundle\Model\User\VisitorInterface;
  11. use Lc\CaracoleBundle\Repository\Merchant\MerchantStore;
  12. use Lc\CaracoleBundle\Repository\MerchantStoreTrait;
  13. use Lc\CaracoleBundle\Repository\Reduction\ReductionCartStore;
  14. use Lc\CaracoleBundle\Repository\Reduction\ReductionCreditStore;
  15. use Lc\CaracoleBundle\Repository\Section\SectionStore;
  16. use Lc\CaracoleBundle\Resolver\OpeningResolver;
  17. use Lc\CaracoleBundle\Repository\SectionStoreTrait;
  18. use Lc\CaracoleBundle\Solver\Order\OrderShopSolver;
  19. use Lc\CaracoleBundle\Solver\Price\PriceSolver;
  20. use Lc\CaracoleBundle\Solver\Reduction\ReductionCartSolver;
  21. use Lc\SovBundle\Model\User\UserInterface;
  22. use Lc\SovBundle\Repository\AbstractStore;
  23. use Lc\SovBundle\Repository\RepositoryQueryInterface;
  24. use Symfony\Component\DependencyInjection\ParameterBag\ParameterBagInterface;
  25. use Symfony\Component\HttpFoundation\Session\Flash\FlashBagInterface;
  26. use Symfony\Component\Routing\Generator\UrlGeneratorInterface;
  27. class OrderShopStore extends AbstractStore
  28. {
  29. use SectionStoreTrait;
  30. use MerchantStoreTrait;
  31. protected OrderShopRepositoryQuery $query;
  32. protected EntityManagerInterface $entityManager;
  33. protected PriceSolver $priceSolver;
  34. protected DocumentBuilder $documentBuilder;
  35. protected ReductionCreditStore $reductionCreditStore;
  36. protected ReductionCartSolver $reductionCartSolver;
  37. protected SectionStore $sectionStore;
  38. protected OrderProductStore $orderProductStore;
  39. protected MerchantStore $merchantStore;
  40. protected FlashBagInterface $flashBag;
  41. protected OpeningResolver $openingResolver;
  42. protected ParameterBagInterface $parameterBag;
  43. protected UrlGeneratorInterface $router;
  44. protected OrderShopSolver $orderShopSolver;
  45. protected ReductionCartStore $reductionCartStore;
  46. public function __construct(
  47. OrderShopRepositoryQuery $query,
  48. EntityManagerInterface $entityManager,
  49. PriceSolver $priceSolver,
  50. DocumentBuilder $documentBuilder,
  51. ReductionCreditStore $reductionCreditStore,
  52. ReductionCartSolver $reductionCartSolver,
  53. SectionStore $sectionStore,
  54. OrderProductStore $orderProductStore,
  55. MerchantStore $merchantStore,
  56. FlashBagInterface $flashBag,
  57. ParameterBagInterface $parameterBag,
  58. UrlGeneratorInterface $router,
  59. OrderShopSolver $orderShopSolver,
  60. ReductionCartStore $reductionCartStore
  61. ) {
  62. $this->query = $query;
  63. $this->entityManager = $entityManager;
  64. $this->priceSolver = $priceSolver;
  65. $this->documentBuilder = $documentBuilder;
  66. $this->reductionCreditStore = $reductionCreditStore;
  67. $this->reductionCartSolver = $reductionCartSolver;
  68. $this->sectionStore = $sectionStore;
  69. $this->orderProductStore = $orderProductStore;
  70. $this->merchantStore = $merchantStore;
  71. $this->flashBag = $flashBag;
  72. $this->parameterBag = $parameterBag;
  73. $this->router = $router;
  74. $this->orderShopSolver = $orderShopSolver;
  75. $this->reductionCartStore = $reductionCartStore;
  76. }
  77. public function orderByDefault(RepositoryQueryInterface $query): RepositoryQueryInterface
  78. {
  79. $query->orderBy('id', 'DESC');
  80. return $query;
  81. }
  82. public function filtersDefault(RepositoryQueryInterface $query): RepositoryQueryInterface
  83. {
  84. $this
  85. ->addFilterBySectionOptionnal($query)
  86. ->addFilterByMerchantViaSectionOptionnal($query);
  87. return $query;
  88. }
  89. public function relationsDefault(RepositoryQueryInterface $query): RepositoryQueryInterface
  90. {
  91. $query->joinOrderProduct();
  92. return $query;
  93. }
  94. // getOrderShopsOfWeek
  95. public function getByCurrentCycle($params = [], $query = null)
  96. {
  97. return $this->getBy(
  98. array_merge(
  99. [
  100. 'cycleNumber' => $this->orderShopSolver->getCycleNumberCurrentOrder($this->section),
  101. 'isValid' => true,
  102. ],
  103. $params
  104. ),
  105. $query
  106. );
  107. }
  108. // getOrderShopsOfWeekByUser
  109. public function getByCurrentCycleAndUser(UserInterface $user = null, array $params = [], $query = null)
  110. {
  111. return $this->getByCurrentCycle(
  112. array_merge(
  113. [
  114. 'user' => $user,
  115. 'cycleNumber' => $this->orderShopSolver->getCycleNumberCurrentOrder($this->section),
  116. 'excludeComplementaryOrderShops' => true
  117. ],
  118. $params
  119. ),
  120. $query
  121. );
  122. }
  123. //public $countOrderShopsOfWeek = null;
  124. // public function countByCurrentCycle(bool $excludeComplementaryOrderShops = true, $query = null)
  125. public function countByCurrentCycle(array $params, $query = null)
  126. {
  127. return $this->countBy(
  128. array_merge(
  129. [
  130. 'cycleNumber' => $this->orderShopSolver->getCycleNumberCurrentOrder($this->section),
  131. 'excludeComplementaryOrderShops' => isset($params['excludeComplementaryOrderShops']) ?? true,
  132. ],
  133. $params
  134. ),
  135. $query
  136. );
  137. // @TODO : optimisation à remettre en place
  138. /*if (is_null($this->countOrderShopsOfWeek)) {
  139. $this->countOrderShopsOfWeek = $this->getByCurrentCycle(
  140. $section,
  141. [
  142. 'count' => true,
  143. 'excludeComplementaryOrderShops' => $excludeComplementaryOrderShops
  144. ]
  145. );
  146. }
  147. return $this->countOrderShopsOfWeek;*/
  148. }
  149. // getNextWeekId
  150. public function getNextCycleId(int $cycleNumber, $query = null): int
  151. {
  152. $lastOrder = $this->getOneLastValidByCycle($cycleNumber, $query);
  153. if ($lastOrder) {
  154. return intval($lastOrder->getCycleId() + 1);
  155. } else {
  156. return 1;
  157. }
  158. }
  159. public function getNextIdValidOrder($query = null): int
  160. {
  161. $lastOrder = $this->getOneLastValid($query);
  162. if ($lastOrder) {
  163. return intval($lastOrder->getIdValidOrder() + 1);
  164. } else {
  165. return 1;
  166. }
  167. }
  168. // countValidOrderShopByUserAllMerchant
  169. public function countValidByUserAllMerchant($user, $query = null): int
  170. {
  171. return $this->countBy(
  172. [
  173. 'user' => $user,
  174. 'isValid' => true,
  175. // @TODO : à tester
  176. 'isMerchantOnline' => true,
  177. 'excludeComplementaryOrderShops' => true
  178. ],
  179. $query
  180. );
  181. }
  182. public function countValidByUser(UserInterface $user = null, $query = null): int
  183. {
  184. return $this->countBy(
  185. [
  186. 'user' => $user,
  187. 'isValid' => true,
  188. 'excludeComplementaryOrderShops' => true
  189. ],
  190. $query
  191. );
  192. }
  193. public function countValidByCurrentCycle($query = null): int
  194. {
  195. return $this->countBy(
  196. [
  197. 'cycleNumber' => $this->orderShopSolver->getCycleNumberCurrentOrder($this->section),
  198. 'isValid' => true,
  199. 'excludeComplementaryOrderShops' => true
  200. ],
  201. $query
  202. );
  203. }
  204. // countValidOrderWithReductionCredit
  205. public function countValidWithReductionCredit(
  206. ReductionCreditInterface $reductionCredit,
  207. UserInterface $user = null,
  208. $query = null
  209. ): int {
  210. $query = $this->createDefaultQuery($query);
  211. if ($user) {
  212. $query->filterByUser($user);
  213. }
  214. $query
  215. ->selectCount()
  216. ->filterByReductionCredit($reductionCredit)
  217. ->filterByStatus(OrderStatusModel::$statusAliasAsValid);
  218. return $query->count();
  219. }
  220. // countValidOrderWithReductionCart
  221. public function countValidWithReductionCart(
  222. ReductionCartInterface $reductionCart,
  223. $query = null
  224. ): int {
  225. $query = $this->createDefaultQuery($query);
  226. $query
  227. ->selectCount()
  228. ->filterByReductionCart($reductionCart)
  229. ->filterByStatus(OrderStatusModel::$statusAliasAsValid);
  230. return $query->count();
  231. }
  232. // countValidOrderWithReductionCartPerUser
  233. public function countValidWithReductionCartByUser(
  234. ReductionCartInterface $reductionCart,
  235. UserInterface $user,
  236. $query = null
  237. ): int {
  238. $query = $this->createDefaultQuery($query);
  239. $query
  240. ->selectCount()
  241. ->filterByUser($user)
  242. ->filterByReductionCart($reductionCart)
  243. ->filterByStatus(OrderStatusModel::$statusAliasAsValid);
  244. return $query->count();
  245. }
  246. // findCartCurrent
  247. public function getOneCartCurrent(UserInterface $user = null, VisitorInterface $visitor = null, $query = null): ?OrderShopInterface
  248. {
  249. $query = $this->createDefaultQuery($query);
  250. if (!is_null($user)) {
  251. $query->filterByUser($user);
  252. }
  253. else {
  254. if (!is_null($visitor)) {
  255. $query->filterByVisitor($visitor);
  256. }
  257. }
  258. $query
  259. ->selectOrderReductionCarts()
  260. ->filterByStatus(OrderStatusModel::$statusAliasAsCart);
  261. return $query->findOne();
  262. }
  263. // findLastOrderValidOfWeek
  264. public function getOneLastValidByCycle(int $cycleNumber, $query = null): ?OrderShopInterface
  265. {
  266. $query = $this->createDefaultQuery($query);
  267. $query
  268. ->filterByCycleNumber($cycleNumber)
  269. ->filterByStatus(OrderStatusModel::$statusAliasAsValid)
  270. ->filterIsNotComplementaryOrderShop()
  271. ->orderBy('.cycleId', 'DESC');
  272. return $query->findOne();
  273. }
  274. //findLastOrderValid
  275. public function getOneLastValid($query = null): ?OrderShopInterface
  276. {
  277. $query = $this->createDefaultQuery($query);
  278. $query
  279. ->filterByStatus(OrderStatusModel::$statusAliasAsValid)
  280. ->filterIsNotComplementaryOrderShop()
  281. ->orderBy('.idValidOrder', 'DESC');
  282. return $query->findOne();
  283. }
  284. public function countBy(array $params = [], $query = null)
  285. {
  286. $query = $this->createDefaultQuery($query);
  287. $query->selectCount();
  288. $this->applyGetByFilters($params, $query);
  289. return $query->count();
  290. }
  291. public function getBy(array $params = [], $query = null): array
  292. {
  293. $query = $this->createDefaultQuery($query);
  294. $this->applyGetByFilters($params, $query);
  295. $orderShops = $query->find();
  296. if (isset($params['mergeComplementaryOrderShops'])) {
  297. foreach ($orderShops as $orderShop) {
  298. $this->orderShopSolver->mergeComplentaryOrderShops($orderShop);
  299. }
  300. }
  301. return $orderShops;
  302. }
  303. protected function applyGetByFilters(array $params, $query)
  304. {
  305. if (isset($params['isMerchantOnline'])) {
  306. $query->filterIsMerchantOnline();
  307. }
  308. if (isset($params['select'])) {
  309. $query->selectParam($params['select']);
  310. }
  311. if (isset($params['dateStart']) || isset($params['dateEnd'])) {
  312. $params['dateField'] = isset($params['dateField']) ? $params['dateField'] : 'validationDate';
  313. }
  314. if (isset($params['dateStart'])) {
  315. $query->filterByDateStart($params['dateField'], $params['dateStart']);
  316. }
  317. if (isset($params['dateEnd'])) {
  318. $query->filterByDateEnd($params['dateField'], $params['dateEnd']);
  319. }
  320. if (isset($params['cycleNumber'])) {
  321. $query->filterByCycleNumber($params['cycleNumber']);
  322. }
  323. if (isset($params['isCart'])) {
  324. $query->filterByStatus(OrderStatusModel::$statusAliasAsCart);
  325. }
  326. if (isset($params['isValid'])) {
  327. $query->filterByStatus(OrderStatusModel::$statusAliasAsValid);
  328. }
  329. if (isset($params['isWaitingDelivery'])) {
  330. $query->filterByStatus(OrderStatusModel::$statusAliasWaitingDelivery);
  331. }
  332. if (isset($params['orderStatus'])) {
  333. $query->filterByStatus([$params['orderStatus']]);
  334. }
  335. if (isset($params['user'])) {
  336. $query->filterByUser($params['user']);
  337. }
  338. if (isset($params['address'])) {
  339. $query->filterByAddress($params['address']);
  340. }
  341. if (isset($params['mergeComplementaryOrderShops'])) {
  342. $query
  343. ->joinComplementaryOrderShops();
  344. }
  345. if (isset($params['excludeComplementaryOrderShops']) || isset($params['mergeComplementaryOrderShops'])) {
  346. $query->filterIsNullMainOrderShop();
  347. }
  348. if (isset($params['orderBy'])) {
  349. $sort = isset($params['orderByDirection']) ? $params['orderByDirection'] : 'DESC';
  350. $query->orderBy($params['orderBy'], $sort);
  351. } else {
  352. $query->orderBy('.id', 'DESC');
  353. }
  354. if (isset($params['groupBy'])) {
  355. $query->groupBy($params['groupBy']);
  356. }
  357. return $query;
  358. }
  359. public function isReductionGiftUsed(ReductionCreditInterface $reductionGift, $query = null)
  360. {
  361. if ($this->countValidWithReductionCredit($reductionGift, null, $query)) {
  362. return true;
  363. } else {
  364. return false;
  365. }
  366. }
  367. public function isReductionCreditUsed(
  368. ReductionCreditInterface $reductionCredit,
  369. UserInterface $user = null,
  370. $query = null
  371. ) {
  372. if ($this->countValidWithReductionCredit($reductionCredit, $user, $query)) {
  373. return true;
  374. } else {
  375. return false;
  376. }
  377. }
  378. public function getReductionCreditsAvailableByUser(UserInterface $user): array
  379. {
  380. $reductionCredits = $this->reductionCreditStore->getByTypeAndUser(ReductionCreditModel::TYPE_CREDIT, $user);
  381. $reductionCreditsArray = [];
  382. foreach ($reductionCredits as $reductionCredit) {
  383. if (!$this->countValidWithReductionCredit($reductionCredit, $user)
  384. && ($reductionCredit->getSection()->getMerchant() == $this->merchant)) {
  385. $reductionCreditsArray[] = $reductionCredit;
  386. }
  387. }
  388. return $reductionCreditsArray;
  389. }
  390. public function getReductionGiftsAvailableByUser($user): array
  391. {
  392. $reductionGifts = $this->reductionCreditStore->getByTypeAndUser(ReductionCreditModel::TYPE_GIFT, $user);
  393. $reductionGiftsArray = [];
  394. foreach ($reductionGifts as $reductionGift) {
  395. if (!$this->countValidWithReductionCredit($reductionGift)) {
  396. $reductionGiftsArray[] = $reductionGift;
  397. }
  398. }
  399. return $reductionGiftsArray;
  400. }
  401. // getReductionCartRemainingQuantity
  402. public function getReductionCartRemainingQuantity(ReductionCartInterface $reductionCart): float
  403. {
  404. return $reductionCart->getAvailableQuantity() - $this->countValidWithReductionCart(
  405. $reductionCart
  406. );
  407. }
  408. // getReductionCartUsedQuantityPerUser
  409. public function getReductionCartUsedQuantityByUser(
  410. ReductionCartInterface $reductionCart,
  411. UserInterface $user
  412. ): float {
  413. return $this->countValidWithReductionCartByUser($reductionCart, $user);
  414. }
  415. // getReductionCartUsedQuantity
  416. public function getReductionCartUsedQuantity(ReductionCartInterface $reductionCart): float
  417. {
  418. return $this->countValidWithReductionCart($reductionCart);
  419. }
  420. // getReductionCartRemainingQuantityPerUser
  421. public function getReductionCartRemainingQuantityByUser(
  422. ReductionCartInterface $reductionCart,
  423. UserInterface $user
  424. ): float {
  425. if ($reductionCart->getAvailableQuantityPerUser()) {
  426. return $reductionCart->getAvailableQuantityPerUser() - $this->countValidWithReductionCartByUser(
  427. $reductionCart,
  428. $user
  429. );
  430. }
  431. return false;
  432. }
  433. // findAllAvailableForUser / getReductionCartsAvailableByUser
  434. public function getReductionCartAvailableByUser(UserInterface $user, $query = null)
  435. {
  436. $reductionCarts = $this->reductionCartStore->getOnline();
  437. $reductionCartsArray = [];
  438. foreach ($reductionCarts as $reductionCart) {
  439. if ($this->reductionCartSolver->matchWithUser($reductionCart, $user)
  440. && $this->reductionCartSolver->matchWithGroupUser($reductionCart, $user)
  441. && $this->getReductionCartRemainingQuantityByUser($reductionCart, $user)
  442. && ($reductionCart->getUsers()->count() > 0 || $reductionCart->getGroupUsers()->count() > 0)
  443. && (!$this->merchant || $reductionCart->getSection()->getMerchant() == $this->merchant)) {
  444. $reductionCartsArray[] = $reductionCart;
  445. }
  446. }
  447. return $reductionCartsArray;
  448. }
  449. public function countValidOrderProductsOfCyclesByProducts(
  450. array $cycleNumbers,
  451. array $products,
  452. $query = null
  453. ): array {
  454. $query = $this->createDefaultQuery($query);
  455. $query
  456. ->filterByAlias(OrderStatusModel::$statusAliasAsValid)
  457. ->filterByCycleNumbers($cycleNumbers)
  458. ->filterByProducts($products)
  459. ->selectSum()
  460. ->groupBy('.cycleNumber, product.id');
  461. return $query->find();
  462. }
  463. public function countValidOrderProductsOfCycleByProduct(int $cycleNumber, int $productId, $query = null): ?string
  464. {
  465. $query = $this->createDefaultQuery($query);
  466. $query
  467. ->filterByAlias(OrderStatusModel::$statusAliasAsValid)
  468. ->filterByCycleNumber($cycleNumber)
  469. ->filterByProduct($productId)
  470. ->selectSumQuantityOrder()
  471. ->groupBy('.cycleNumber, product.id');
  472. $result = $query->findOne();
  473. if ($result) {
  474. return $result['quantity'];
  475. }
  476. return null;
  477. }
  478. public function isReductionCreditAllowAddToOrder(
  479. OrderShopInterface $orderShop,
  480. ReductionCreditInterface $reductionCredit
  481. ) {
  482. $user = $orderShop->getUser();
  483. // appartient à l'utilisateur
  484. if (!$reductionCredit->getUsers()->contains($user)) {
  485. // @TODO : déplacer la gestion du flash message
  486. //$this->flashBag->add('error', 'error.reductionCredit.userNotAllow');
  487. return false;
  488. }
  489. // n'a pas été utilisé
  490. if ($reductionCredit->getType() == ReductionCreditModel::TYPE_CREDIT) {
  491. if ($this->countValidWithReductionCredit($reductionCredit, $user) > 0) {
  492. // @TODO : déplacer la gestion du flash message
  493. //$this->flashBah->add('error', 'error.reductionCredit.alreadyUse');
  494. return false;
  495. }
  496. } else {
  497. if ($this->countValidWithReductionCredit($reductionCredit) > 0) {
  498. // @TODO : déplacer la gestion du flash message
  499. //$this->flashBah->add('error', 'error.reductionCredit.alreadyUse');
  500. return false;
  501. }
  502. }
  503. return true;
  504. }
  505. }