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.

ProductController.php 19KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531
  1. <?php
  2. /**
  3. * Copyright distrib (2018)
  4. *
  5. * contact@opendistrib.net
  6. *
  7. * Ce logiciel est un programme informatique servant à aider les producteurs
  8. * à distribuer leur production en circuits courts.
  9. *
  10. * Ce logiciel est régi par la licence CeCILL soumise au droit français et
  11. * respectant les principes de diffusion des logiciels libres. Vous pouvez
  12. * utiliser, modifier et/ou redistribuer ce programme sous les conditions
  13. * de la licence CeCILL telle que diffusée par le CEA, le CNRS et l'INRIA
  14. * sur le site "http://www.cecill.info".
  15. *
  16. * En contrepartie de l'accessibilité au code source et des droits de copie,
  17. * de modification et de redistribution accordés par cette licence, il n'est
  18. * offert aux utilisateurs qu'une garantie limitée. Pour les mêmes raisons,
  19. * seule une responsabilité restreinte pèse sur l'auteur du programme, le
  20. * titulaire des droits patrimoniaux et les concédants successifs.
  21. *
  22. * A cet égard l'attention de l'utilisateur est attirée sur les risques
  23. * associés au chargement, à l'utilisation, à la modification et/ou au
  24. * développement et à la reproduction du logiciel par l'utilisateur étant
  25. * donné sa spécificité de logiciel libre, qui peut le rendre complexe à
  26. * manipuler et qui le réserve donc à des développeurs et des professionnels
  27. * avertis possédant des connaissances informatiques approfondies. Les
  28. * utilisateurs sont donc invités à charger et tester l'adéquation du
  29. * logiciel à leurs besoins dans des conditions permettant d'assurer la
  30. * sécurité de leurs systèmes et ou de leurs données et, plus généralement,
  31. * à l'utiliser et l'exploiter dans les mêmes conditions de sécurité.
  32. *
  33. * Le fait que vous puissiez accéder à cet en-tête signifie que vous avez
  34. * pris connaissance de la licence CeCILL, et que vous en avez accepté les
  35. * termes.
  36. */
  37. namespace backend\controllers;
  38. use backend\forms\ProductPriceUploadForm;
  39. use common\helpers\CSV;
  40. use common\helpers\GlobalParam;
  41. use common\logic\Distribution\ProductDistribution\Model\ProductDistribution;
  42. use common\logic\PointSale\PointSale\Model\PointSale;
  43. use common\logic\Product\Product\Model\Product;
  44. use common\logic\Product\Product\Model\ProductSearch;
  45. use common\logic\Product\ProductPointSale\Model\ProductPointSale;
  46. use common\logic\Product\ProductPrice\Model\ProductPrice;
  47. use common\logic\Product\ProductPrice\Model\ProductPriceSearch;
  48. use common\logic\User\UserProducer\Model\UserProducer;
  49. use Yii;
  50. use yii\filters\AccessControl;
  51. use yii\helpers\BaseFileHelper;
  52. use yii\helpers\Html;
  53. use yii\web\NotFoundHttpException;
  54. use yii\filters\VerbFilter;
  55. use common\helpers\Upload;
  56. use yii\web\UploadedFile;
  57. /**
  58. * ProduitController implements the CRUD actions for Produit model.
  59. */
  60. class ProductController extends BackendController
  61. {
  62. var $enableCsrfValidation = false;
  63. public function behaviors()
  64. {
  65. return [
  66. 'verbs' => [
  67. 'class' => VerbFilter::class,
  68. 'actions' => [
  69. ],
  70. ],
  71. 'access' => [
  72. 'class' => AccessControl::class,
  73. 'rules' => [
  74. [
  75. 'allow' => true,
  76. 'roles' => ['@'],
  77. 'matchCallback' => function ($rule, $action) {
  78. return $this->getUserManager()->hasAccessBackend();
  79. }
  80. ]
  81. ],
  82. ],
  83. ];
  84. }
  85. /**
  86. * Liste les modèles Produit.
  87. *
  88. * @return mixed
  89. */
  90. public function actionIndex()
  91. {
  92. $searchModel = new ProductSearch();
  93. $dataProvider = $searchModel->search(\Yii::$app->request->queryParams);
  94. return $this->render('index', [
  95. 'searchModel' => $searchModel,
  96. 'dataProvider' => $dataProvider,
  97. ]);
  98. }
  99. /**
  100. * Crée un Product.
  101. */
  102. public function actionCreate()
  103. {
  104. $productManager = $this->getProductManager();
  105. $distributionManager = $this->getDistributionManager();
  106. $model = $productManager->instanciateProduct();
  107. $model->active = 1;
  108. $model->id_producer = GlobalParam::getCurrentProducerId();
  109. $model->monday = 1;
  110. $model->tuesday = 1;
  111. $model->wednesday = 1;
  112. $model->thursday = 1;
  113. $model->friday = 1;
  114. $model->saturday = 1;
  115. $model->sunday = 1;
  116. $model->available_on_points_sale = 1;
  117. if ($model->load(\Yii::$app->request->post()) && $productManager->saveCreate($model)) {
  118. $lastProductOrder = Product::find()->where('id_producer = :id_producer')->params([':id_producer' => GlobalParam::getCurrentProducerId()])->orderBy('order DESC')->one();
  119. if ($lastProductOrder) {
  120. $model->order = ++$lastProductOrder->order;
  121. }
  122. Upload::uploadFile($model, 'photo');
  123. $productManager->saveUpdate($model);
  124. $this->processAvailabilityPointsSale($model);
  125. $distributionManager->addProductIncomingDistributions($model);
  126. $this->setFlash('success', 'Produit <strong>' . Html::encode($model->name) . '</strong> ajouté');
  127. return $this->redirect(['index']);
  128. } else {
  129. return $this->render('create', [
  130. 'model' => $model,
  131. ]);
  132. }
  133. }
  134. /**
  135. * Modifie un Product.
  136. */
  137. public function actionUpdate($id)
  138. {
  139. $productManager = $this->getProductManager();
  140. $distributionManager = $this->getDistributionManager();
  141. $request = Yii::$app->request;
  142. $model = $this->findModel($id);
  143. foreach ($model->productPointSale as $productPointSale) {
  144. $model->pointsSale[] = $productPointSale->id_point_sale;
  145. }
  146. $photoFilenameOld = $model->photo;
  147. if ($model->load(\Yii::$app->request->post()) && $productManager->saveUpdate($model)) {
  148. Upload::uploadFile($model, 'photo', $photoFilenameOld);
  149. $deletePhoto = $request->post('delete_photo', 0);
  150. if ($deletePhoto) {
  151. $model->photo = '';
  152. $model->save();
  153. }
  154. $this->processAvailabilityPointsSale($model);
  155. if ($model->apply_distributions) {
  156. $distributionManager->addProductIncomingDistributions($model);
  157. }
  158. $this->setFlash('success', 'Produit <strong>' . Html::encode($model->name) . '</strong> modifié');
  159. return $this->redirect(['index']);
  160. }
  161. return $this->render('update/update', [
  162. 'model' => $model,
  163. 'action' => 'update',
  164. ]);
  165. }
  166. /**
  167. * Traite les accès restreints d'un point de vente.
  168. */
  169. public function processAvailabilityPointsSale($product)
  170. {
  171. $pointSaleManager = $this->getPointSaleManager();
  172. $productPointSaleManager = $this->getProductPointSaleManager();
  173. ProductPointSale::deleteAll(['id_product' => $product->id]);
  174. if (is_array($product->pointsSale) && count($product->pointsSale)) {
  175. foreach ($product->pointsSale as $key => $idPointSale) {
  176. $pointSale = $pointSaleManager->findOnePointSaleById($idPointSale);
  177. if ($pointSale) {
  178. $productPointSaleManager->createProductPointSale(
  179. $product,
  180. $pointSale,
  181. ($product->available_on_points_sale) ? false : true
  182. );
  183. }
  184. }
  185. }
  186. }
  187. public function actionPricesList(int $id)
  188. {
  189. $model = $this->findModel($id);
  190. $searchModel = new ProductPriceSearch();
  191. $searchModel->id_product = $id;
  192. $dataProvider = $searchModel->search(array_merge(\Yii::$app->request->queryParams, [
  193. 'id_product' => $id
  194. ]));
  195. $userProducerWithProductPercent = UserProducer::searchAll([], [
  196. 'join_with' => ['user'],
  197. 'conditions' => 'user_producer.product_price_percent != 0',
  198. ]);
  199. $pointSaleWithProductPercent = PointSale::searchAll([], [
  200. 'conditions' => 'point_sale.product_price_percent != 0'
  201. ]);
  202. return $this->render('update/prices/list', [
  203. 'model' => $model,
  204. 'action' => 'prices-list',
  205. 'searchModel' => $searchModel,
  206. 'dataProvider' => $dataProvider,
  207. 'userProducerWithProductPercent' => $userProducerWithProductPercent,
  208. 'pointSaleWithProductPercent' => $pointSaleWithProductPercent,
  209. ]);
  210. }
  211. public function actionPricesCreate($idProduct)
  212. {
  213. $model = new ProductPrice();
  214. $model->id_product = $idProduct;
  215. $modelProduct = $this->findModel($idProduct);
  216. if ($model->load(\Yii::$app->request->post())) {
  217. $conditionsProductPriceExist = [
  218. 'id_product' => $idProduct,
  219. 'id_user' => $model->id_user ?? null,
  220. 'id_user_group' => $model->id_user_group ?? null,
  221. 'id_point_sale' => $model->id_point_sale ?? null,
  222. 'from_quantity' => $model->from_quantity ?? null,
  223. ];
  224. $productPriceExist = ProductPrice::findOne($conditionsProductPriceExist);
  225. if ($productPriceExist) {
  226. $productPriceExist->delete();
  227. $this->setFlash('warning', 'Un prix existait déjà pour cet utilisateur / point de vente, il a été supprimé.');
  228. }
  229. if ($model->save()) {
  230. $this->setFlash('success', 'Le prix a bien été ajouté.');
  231. return $this->redirect(['product/prices-list', 'id' => $idProduct]);
  232. }
  233. }
  234. return $this->render('update/prices/create', [
  235. 'model' => $model,
  236. 'modelProduct' => $modelProduct,
  237. ]);
  238. }
  239. public function actionPricesUpdate($id)
  240. {
  241. $model = $this->findModelProductPrice($id);
  242. $modelProduct = $this->findModel($model->id_product);
  243. if ($model->load(\Yii::$app->request->post()) && $model->save()) {
  244. $this->setFlash('success', 'Prix modifié');
  245. return $this->redirect(['product/prices-list', 'id' => $model->id_product]);
  246. }
  247. return $this->render('update/prices/update', [
  248. 'model' => $model,
  249. 'modelProduct' => $modelProduct,
  250. 'action' => 'prices-update',
  251. ]);
  252. }
  253. public function actionPricesDelete($id)
  254. {
  255. $productPrice = $this->findModelProductPrice($id);
  256. $productPrice->delete();
  257. $this->setFlash('success', 'Prix supprimé');
  258. return $this->redirect(['product/prices-list', 'id' => $productPrice->id_product]);
  259. }
  260. /**
  261. * Supprime un Product.
  262. */
  263. public function actionDelete(int $id, bool $confirm = false)
  264. {
  265. $product = $this->findModel($id);
  266. if ($confirm) {
  267. $product->delete();
  268. ProductDistribution::deleteAll(['id_product' => $id]);
  269. $this->setFlash('success', 'Produit <strong>' . Html::encode($product->name) . '</strong> supprimé');
  270. } else {
  271. $this->setFlash('info', 'Souhaitez-vous vraiment supprimer le produit <strong>' . Html::encode($product->name) . '</strong> ? '
  272. . Html::a('Oui', ['product/delete', 'id' => $id, 'confirm' => 1], ['class' => 'btn btn-default']) . ' ' . Html::a('Non', ['product/index'], ['class' => 'btn btn-default']));
  273. }
  274. return $this->redirect(['index']);
  275. }
  276. /**
  277. * Modifie l'ordre des produits.
  278. */
  279. public function actionOrder()
  280. {
  281. $array = Yii::$app->request->post('array');
  282. $orderArray = json_decode(stripslashes($array));
  283. foreach ($orderArray as $id => $order) {
  284. $product = $this->findModel($id);
  285. $product->order = $order;
  286. $product->save();
  287. }
  288. }
  289. public function actionAjaxToggleActive($id, $active)
  290. {
  291. \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
  292. $distributionManager = $this->getDistributionManager();
  293. $product = $this->findModel($id);
  294. $product->active = (int) $active;
  295. $product->save();
  296. $distributionManager->addProductIncomingDistributions($product);
  297. return ['success', 'id' => $id, 'active' => $active];
  298. }
  299. /**
  300. * Import des prix produits via un fichier CSV.
  301. *
  302. * @return mixed
  303. */
  304. public function actionPriceImport()
  305. {
  306. $productManager = $this->getProductManager();
  307. $productPriceManager = $this->getProductPriceManager();
  308. $userGroupManager = $this->getUserGroupManager();
  309. $pointSaleManager = $this->getPointSaleManager();
  310. $userManager = $this->getUserManager();
  311. $model = new ProductPriceUploadForm();
  312. if (Yii::$app->request->isPost) {
  313. $model->file = UploadedFile::getInstance($model, 'file');
  314. if ($model->file && $model->validate()) {
  315. $productPriceCsvArray = array_map(function($data) { return str_getcsv($data,";");}, file($model->file->tempName));
  316. //die('mime type : '.BaseFileHelper::getMimeType($model->file->tempName));
  317. if(!$productPriceCsvArray || count($productPriceCsvArray[0]) != 6) {
  318. $this->setFlash('error', "Format de fichier invalide. Veuillez vérifier que le séparateur de champs de votre fichier est bien \";\".");
  319. }
  320. else {
  321. unset($productPriceCsvArray[0]);
  322. $countUpdate = 0;
  323. $dataNotFound = false;
  324. foreach ($productPriceCsvArray as $productPriceCsv) {
  325. if (count($productPriceCsv) != 6) {
  326. $dataNotFound = true;
  327. continue;
  328. }
  329. $productName = $productPriceCsv[0];
  330. $userArray = explode('#', $productPriceCsv[1]);
  331. $userGroupName = $productPriceCsv[2];
  332. $pointSaleName = $productPriceCsv[3];
  333. $quantityFrom = (float)$productPriceCsv[4];
  334. $price = (float)$productPriceCsv[5];
  335. $product = $productName ? $productManager->findOneProductByName($productName) : null;
  336. $user = (count($userArray) > 1) ? $userManager->findOneUserById((int)$userArray[1]) : null;
  337. $userGroup = $userGroupName ? $userGroupManager->findOneUserGroupByName($userGroupName) : null;
  338. $pointSale = $pointSaleName ? $pointSaleManager->findOnePointSaleByName($pointSaleName) : null;
  339. if (($productName && !$product)
  340. || (count($userArray) > 1 && !$user)
  341. || ($userGroupName && !$userGroup)
  342. || ($pointSaleName && !$pointSale)) {
  343. $dataNotFound = true;
  344. continue;
  345. }
  346. if ($product) {
  347. // prix de base
  348. if (!$user && !$userGroup && !$pointSale && !$quantityFrom) {
  349. $product->price = $price;
  350. $productManager->saveUpdate($product);
  351. $countUpdate++;
  352. } // prix spécifique
  353. else {
  354. $productPrice = $productPriceManager->findOneProductPriceBy($product, $user, $userGroup, $pointSale, $quantityFrom);
  355. if ($productPrice) {
  356. $productPrice->price = $price;
  357. $productPriceManager->saveUpdate($productPrice);
  358. $countUpdate++;
  359. } else {
  360. $dataNotFound = true;
  361. }
  362. }
  363. }
  364. }
  365. if ($dataNotFound) {
  366. $this->addFlash('error', "Attention, certaines lignes du fichier n'ont pas été prises en compte. Veuillez réessayer en repartant du fichier d'export.<br />Contacter l'administrateur du site si le problème persiste.");
  367. }
  368. if ($countUpdate) {
  369. $this->addFlash('success', $countUpdate . ' prix produits mis à jour.');
  370. }
  371. }
  372. }
  373. }
  374. return $this->render('price_import', [
  375. 'model' => $model
  376. ]);
  377. }
  378. /**
  379. * Export des prix produits au format CSV.
  380. *
  381. * @return mixed
  382. */
  383. public function actionPriceExport()
  384. {
  385. $productManager = $this->getProductManager();
  386. $productPriceManager = $this->getProductPriceManager();
  387. $userManager = $this->getUserManager();
  388. $data = [];
  389. $data[] = [
  390. "Produit",
  391. "Utilisateur",
  392. "Groupe d'utilisateur",
  393. "Point de vente",
  394. "À partir de la quantité",
  395. "Prix HT"
  396. ];
  397. $productArray = $productManager->findProducts();
  398. foreach($productArray as $product) {
  399. // prix produit
  400. $data[] = [
  401. $product->name,
  402. '',
  403. '',
  404. '',
  405. '',
  406. $product->price
  407. ];
  408. // prix spécifiques
  409. foreach($product->productPrice as $productPrice) {
  410. $productPrice = $productPriceManager->findOneProductPriceById($productPrice->id);
  411. if($productPrice->user || $productPrice->userGroup || $productPrice->pointSale || $productPrice->from_quantity) {
  412. $data[] = [
  413. $product->name,
  414. $productPrice->user ? str_replace('#', '', $userManager->getUsername($productPrice->user)).' #'.$productPrice->user->id : '',
  415. $productPrice->userGroup ? $productPrice->userGroup->name : '',
  416. $productPrice->pointSale ? $productPrice->pointSale->name : '',
  417. $productPrice->from_quantity ?? '',
  418. $productPrice->price
  419. ];
  420. }
  421. }
  422. }
  423. CSV::downloadSendHeaders('prix_produits.csv');
  424. echo CSV::array2csv($data);
  425. die();
  426. }
  427. /**
  428. * Recherche un produit en fonction de son ID.
  429. */
  430. protected function findModel(int $id)
  431. {
  432. $productManager = $this->getProductManager();
  433. if (($product = $productManager->findOneProductById($id)) !== null) {
  434. return $product;
  435. } else {
  436. throw new NotFoundHttpException('The requested page does not exist.');
  437. }
  438. }
  439. protected function findModelProductPrice($id)
  440. {
  441. $productPriceManager = $this->getProductPriceManager();
  442. if (($productPrice = $productPriceManager->findOneProductPriceById($id)) !== null) {
  443. return $productPrice;
  444. } else {
  445. throw new NotFoundHttpException('The requested page does not exist.');
  446. }
  447. }
  448. }