No puede seleccionar más de 25 temas Los temas deben comenzar con una letra o número, pueden incluir guiones ('-') y pueden tener hasta 35 caracteres de largo.

565 líneas
21KB

  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\Feature\Feature\Model\Feature;
  43. use common\logic\PointSale\PointSale\Model\PointSale;
  44. use common\logic\Product\Product\Model\Product;
  45. use common\logic\Product\Product\Model\ProductSearch;
  46. use common\logic\Product\ProductPointSale\Model\ProductPointSale;
  47. use common\logic\Product\ProductPrice\Model\ProductPrice;
  48. use common\logic\Product\ProductPrice\Model\ProductPriceSearch;
  49. use common\logic\User\UserProducer\Model\UserProducer;
  50. use Yii;
  51. use yii\filters\AccessControl;
  52. use yii\helpers\BaseFileHelper;
  53. use yii\helpers\Html;
  54. use yii\web\NotFoundHttpException;
  55. use yii\filters\VerbFilter;
  56. use common\helpers\Upload;
  57. use yii\web\UploadedFile;
  58. /**
  59. * ProduitController implements the CRUD actions for Produit model.
  60. */
  61. class ProductController extends BackendController
  62. {
  63. var $enableCsrfValidation = false;
  64. public function behaviors()
  65. {
  66. return [
  67. 'access' => [
  68. 'class' => AccessControl::class,
  69. 'rules' => [
  70. [
  71. 'allow' => true,
  72. 'roles' => ['@'],
  73. 'matchCallback' => function ($rule, $action) {
  74. return $this->getUserModule()
  75. ->getAuthorizationChecker()
  76. ->isGrantedAsProducer($this->getUserCurrent());
  77. }
  78. ]
  79. ],
  80. ],
  81. ];
  82. }
  83. /**
  84. * Liste les modèles Produit.
  85. *
  86. * @return mixed
  87. */
  88. public function actionIndex()
  89. {
  90. $searchModel = new ProductSearch();
  91. $dataProvider = $searchModel->search(\Yii::$app->request->queryParams);
  92. return $this->render('index', [
  93. 'searchModel' => $searchModel,
  94. 'dataProvider' => $dataProvider,
  95. ]);
  96. }
  97. /**
  98. * Crée un Product.
  99. */
  100. public function actionCreate()
  101. {
  102. $productModule = $this->getProductModule();
  103. $distributionModule = $this-> getDistributionModule();
  104. $model = $productModule->instanciateProduct();
  105. $model->status = Product::STATUS_ONLINE;
  106. $model->id_producer = GlobalParam::getCurrentProducerId();
  107. $model->monday = 1;
  108. $model->tuesday = 1;
  109. $model->wednesday = 1;
  110. $model->thursday = 1;
  111. $model->friday = 1;
  112. $model->saturday = 1;
  113. $model->sunday = 1;
  114. $model->available_on_points_sale = 1;
  115. if ($model->load(\Yii::$app->request->post())) {
  116. $model->photoFile = UploadedFile::getInstance($model, 'photoFile');
  117. if($model->validate()) {
  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. $productModule->create($model);
  123. if($model->photoFile) {
  124. Upload::uploadFile($model, 'photoFile', 'photo');
  125. }
  126. $this->processAvailabilityPointsSale($model);
  127. $distributionModule->addProductIncomingDistributions($model);
  128. $this->setFlash('success', 'Produit <strong>' . Html::encode($model->name) . '</strong> ajouté');
  129. return $this->redirect(['index']);
  130. }
  131. }
  132. return $this->render('create', [
  133. 'model' => $model,
  134. ]);
  135. }
  136. /**
  137. * Modifie un Product.
  138. */
  139. public function actionUpdate($id)
  140. {
  141. $productModule = $this->getProductModule();
  142. $distributionModule = $this-> getDistributionModule();
  143. $request = Yii::$app->request;
  144. $model = $this->findModel($id);
  145. foreach ($model->productPointSale as $productPointSale) {
  146. $model->pointsSale[] = $productPointSale->id_point_sale;
  147. }
  148. $photoFilenameOld = $model->photo;
  149. if (Yii::$app->request->isPost && $model->load(\Yii::$app->request->post())) {
  150. $model->photoFile = UploadedFile::getInstance($model, 'photoFile');
  151. if($model->validate()) {
  152. if($model->photoFile) {
  153. Upload::uploadFile($model, 'photoFile', 'photo', $photoFilenameOld);
  154. }
  155. $deletePhoto = $request->post('delete_photo', 0);
  156. if ($deletePhoto) {
  157. $model->photo = '';
  158. $model->save();
  159. }
  160. $this->processAvailabilityPointsSale($model);
  161. if ($model->apply_distributions) {
  162. $distributionModule->addProductIncomingDistributions($model);
  163. }
  164. $productModule->update($model);
  165. $this->setFlash('success', 'Produit <strong>' . Html::encode($model->name) . '</strong> modifié');
  166. return $this->redirect(['index']);
  167. }
  168. }
  169. return $this->render('update/update', [
  170. 'model' => $model,
  171. 'action' => 'update',
  172. ]);
  173. }
  174. /**
  175. * Traite les accès restreints d'un point de vente.
  176. */
  177. public function processAvailabilityPointsSale($product)
  178. {
  179. $pointSaleModule = $this->getPointSaleModule();
  180. $productPointSaleModule = $this->getProductPointSaleModule();
  181. ProductPointSale::deleteAll(['id_product' => $product->id]);
  182. if (is_array($product->pointsSale) && count($product->pointsSale)) {
  183. foreach ($product->pointsSale as $key => $idPointSale) {
  184. $pointSale = $pointSaleModule->findOnePointSaleById($idPointSale);
  185. if ($pointSale) {
  186. $productPointSaleModule->createProductPointSale(
  187. $product,
  188. $pointSale,
  189. ($product->available_on_points_sale) ? false : true
  190. );
  191. }
  192. }
  193. }
  194. }
  195. public function actionPricesList(int $id)
  196. {
  197. $model = $this->findModel($id);
  198. $searchModel = new ProductPriceSearch();
  199. $searchModel->id_product = $id;
  200. $dataProvider = $searchModel->search(array_merge(\Yii::$app->request->queryParams, [
  201. 'id_product' => $id
  202. ]));
  203. $userProducerWithProductPercent = UserProducer::searchAll([], [
  204. 'join_with' => ['user'],
  205. 'conditions' => 'user_producer.product_price_percent != 0',
  206. ]);
  207. $pointSaleWithProductPercent = PointSale::searchAll([], [
  208. 'conditions' => 'point_sale.product_price_percent != 0'
  209. ]);
  210. return $this->render('update/prices/list', [
  211. 'model' => $model,
  212. 'action' => 'prices-list',
  213. 'searchModel' => $searchModel,
  214. 'dataProvider' => $dataProvider,
  215. 'userProducerWithProductPercent' => $userProducerWithProductPercent,
  216. 'pointSaleWithProductPercent' => $pointSaleWithProductPercent,
  217. ]);
  218. }
  219. public function actionPricesCreate($idProduct)
  220. {
  221. $model = new ProductPrice();
  222. $model->id_product = $idProduct;
  223. $modelProduct = $this->findModel($idProduct);
  224. if ($model->load(\Yii::$app->request->post())) {
  225. $conditionsProductPriceExist = [
  226. 'id_product' => $idProduct,
  227. 'id_user' => $model->id_user ?? null,
  228. 'id_user_group' => $model->id_user_group ?? null,
  229. 'id_point_sale' => $model->id_point_sale ?? null,
  230. 'from_quantity' => $model->from_quantity ?? null,
  231. ];
  232. $productPriceExist = ProductPrice::findOne($conditionsProductPriceExist);
  233. if ($productPriceExist) {
  234. $productPriceExist->delete();
  235. $this->setFlash('warning', 'Un prix existait déjà pour cet utilisateur / point de vente, il a été supprimé.');
  236. }
  237. if ($model->save()) {
  238. $this->setFlash('success', 'Le prix a bien été ajouté.');
  239. return $this->redirect(['product/prices-list', 'id' => $idProduct]);
  240. }
  241. }
  242. return $this->render('update/prices/create', [
  243. 'model' => $model,
  244. 'modelProduct' => $modelProduct,
  245. ]);
  246. }
  247. public function actionPricesUpdate($id)
  248. {
  249. $model = $this->findModelProductPrice($id);
  250. $modelProduct = $this->findModel($model->id_product);
  251. if ($model->load(\Yii::$app->request->post()) && $model->save()) {
  252. $this->setFlash('success', 'Prix modifié');
  253. return $this->redirect(['product/prices-list', 'id' => $model->id_product]);
  254. }
  255. return $this->render('update/prices/update', [
  256. 'model' => $model,
  257. 'modelProduct' => $modelProduct,
  258. 'action' => 'prices-update',
  259. ]);
  260. }
  261. public function actionPricesDelete($id)
  262. {
  263. $productPrice = $this->findModelProductPrice($id);
  264. $productPrice->delete();
  265. $this->setFlash('success', 'Prix supprimé');
  266. return $this->redirect(['product/prices-list', 'id' => $productPrice->id_product]);
  267. }
  268. /**
  269. * Supprime un Product.
  270. */
  271. public function actionDelete(int $id, bool $confirm = false)
  272. {
  273. $productModule = $this->getProductModule();
  274. $productDistributionModule = $this->getProductDistributionModule();
  275. $product = $this->findModel($id);
  276. if ($confirm) {
  277. $productModule->getBuilder()->updateStatusDeleted($product);
  278. $productDistributionModule->getBuilder()->disableProductDistributionsIncomingByProduct($product);
  279. $this->setFlash('success', 'Produit <strong>' . Html::encode($product->name) . '</strong> supprimé');
  280. } else {
  281. $this->setFlash('info', 'Souhaitez-vous vraiment supprimer le produit <strong>' . Html::encode($product->name) . '</strong> ? '
  282. . Html::a('Oui', ['product/delete', 'id' => $id, 'confirm' => 1], ['class' => 'btn btn-default']) . ' ' . Html::a('Non', ['product/index'], ['class' => 'btn btn-default']));
  283. }
  284. return $this->redirect(['index']);
  285. }
  286. /**
  287. * Modifie l'ordre des produits.
  288. */
  289. public function actionOrder()
  290. {
  291. $array = Yii::$app->request->post('array');
  292. $orderArray = json_decode(stripslashes($array));
  293. foreach ($orderArray as $id => $order) {
  294. $product = $this->findModel($id);
  295. $product->order = $order;
  296. $product->save();
  297. }
  298. }
  299. public function actionAjaxToggleStatus($id, $status)
  300. {
  301. \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
  302. $distributionModule = $this-> getDistributionModule();
  303. $product = $this->findModel($id);
  304. $product->status = (int) $status;
  305. $product->save();
  306. $distributionModule->addProductIncomingDistributions($product);
  307. return ['success', 'id' => $id, 'status' => $status];
  308. }
  309. /**
  310. * Import des prix produits via un fichier CSV.
  311. *
  312. * @return mixed
  313. */
  314. public function actionPriceImport()
  315. {
  316. if($this->getFeatureModule()->getManager()->isDisabled(Feature::ALIAS_PRODUCT_PRICE_IMPORT)) {
  317. return $this->redirectDashboard();
  318. }
  319. $productModule = $this->getProductModule();
  320. $productPriceModule = $this->getProductPriceModule();
  321. $userGroupModule = $this->getUserGroupModule();
  322. $pointSaleModule = $this->getPointSaleModule();
  323. $userModule = $this->getUserModule();
  324. $model = new ProductPriceUploadForm();
  325. if (Yii::$app->request->isPost) {
  326. $model->file = UploadedFile::getInstance($model, 'file');
  327. if ($model->file && $model->validate()) {
  328. $productPriceCsvArray = array_map(function($data) { return str_getcsv($data,";");}, file($model->file->tempName));
  329. if(!$productPriceCsvArray || count($productPriceCsvArray[0]) != 6) {
  330. $this->setFlash('error', "Format de fichier invalide. Veuillez vérifier que le séparateur de champs de votre fichier est bien \";\".");
  331. }
  332. else {
  333. unset($productPriceCsvArray[0]);
  334. $countUpdate = 0;
  335. $countCreate = 0;
  336. $cptLine = 1;
  337. $dataNotFound = false;
  338. $dataNotFoundArray = [];
  339. foreach ($productPriceCsvArray as $productPriceCsv) {
  340. $cptLine ++;
  341. if (count($productPriceCsv) != 6) {
  342. $dataNotFound = true;
  343. continue;
  344. }
  345. $productName = $productPriceCsv[0];
  346. $userArray = explode('#', $productPriceCsv[1]);
  347. $userGroupName = $productPriceCsv[2];
  348. $pointSaleName = $productPriceCsv[3];
  349. $quantityFrom = (float)$productPriceCsv[4];
  350. $price = (float) str_replace(',', '.', $productPriceCsv[5]);
  351. $product = $productName ? $productModule->findOneProductByName($productName) : null;
  352. $user = (count($userArray) > 1) ? $userModule->findOneUserById((int)$userArray[1]) : null;
  353. $userGroup = $userGroupName ? $userGroupModule->findOneUserGroupByName($userGroupName) : null;
  354. $pointSale = $pointSaleName ? $pointSaleModule->findOnePointSaleByName($pointSaleName) : null;
  355. if (($productName && !$product)
  356. || (count($userArray) > 1 && !$user)
  357. || ($userGroupName && !$userGroup)
  358. || ($pointSaleName && !$pointSale)) {
  359. $dataNotFound = true;
  360. $dataNotFoundArray[] = $cptLine;
  361. continue;
  362. }
  363. if ($product) {
  364. // prix de base
  365. if (!$user && !$userGroup && !$pointSale && !$quantityFrom) {
  366. $product->price = $price;
  367. $productModule->saveUpdate($product);
  368. $countUpdate++;
  369. } // prix spécifique
  370. else {
  371. $productPrice = $productPriceModule->findOneProductPriceBy($product, $user, $userGroup, $pointSale, $quantityFrom);
  372. if ($productPrice) {
  373. $productPrice->price = $price;
  374. $productPriceModule->saveUpdate($productPrice);
  375. $countUpdate++;
  376. }
  377. // Création automatique du prix spécifique
  378. else {
  379. $productPrice = $productPriceModule->instanciateProductPrice($product, $price, $user, $userGroup, $pointSale, $quantityFrom);
  380. $productPriceModule->saveCreate($productPrice);
  381. $countCreate ++;
  382. }
  383. }
  384. }
  385. }
  386. if ($dataNotFound) {
  387. $strLinesDataNotFound = '('.implode(', ', $dataNotFoundArray).')';
  388. $this->addFlash('error', "Attention, certaines lignes ".$strLinesDataNotFound." 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.");
  389. }
  390. if ($countUpdate) {
  391. $this->addFlash('success', $countUpdate . ' prix produits mis à jour.');
  392. }
  393. if($countCreate) {
  394. $this->addFlash('success', $countCreate . ' prix produits créés.');
  395. }
  396. }
  397. }
  398. }
  399. return $this->render('price_import', [
  400. 'model' => $model
  401. ]);
  402. }
  403. /**
  404. * Export des prix produits au format CSV.
  405. *
  406. * @return mixed
  407. */
  408. public function actionPriceExport()
  409. {
  410. if($this->getFeatureModule()->getManager()->isDisabled(Feature::ALIAS_PRODUCT_PRICE_IMPORT)) {
  411. return $this->redirectDashboard();
  412. }
  413. $productModule = $this->getProductModule();
  414. $productPriceModule = $this->getProductPriceModule();
  415. $userModule = $this->getUserModule();
  416. $data = [];
  417. $data[] = [
  418. "Produit",
  419. "Utilisateur",
  420. "Groupe d'utilisateur",
  421. "Point de vente",
  422. "À partir de la quantité",
  423. "Prix HT"
  424. ];
  425. $productArray = $productModule->findProducts();
  426. foreach($productArray as $product) {
  427. // prix produit
  428. $data[] = [
  429. $product->name,
  430. '',
  431. '',
  432. '',
  433. '',
  434. CSV::formatNumber($product->price)
  435. ];
  436. // prix spécifiques
  437. foreach($product->productPrice as $productPrice) {
  438. $productPrice = $productPriceModule->findOneProductPriceById($productPrice->id);
  439. if($productPrice->user || $productPrice->userGroup || $productPrice->pointSale || $productPrice->from_quantity) {
  440. $data[] = [
  441. $product->name,
  442. $productPrice->user ? str_replace('#', '', $userModule->getUsername($productPrice->user)).' #'.$productPrice->user->id : '',
  443. $productPrice->userGroup ? $productPrice->userGroup->name : '',
  444. $productPrice->pointSale ? $productPrice->pointSale->name : '',
  445. $productPrice->from_quantity ?? '',
  446. CSV::formatNumber($productPrice->price)
  447. ];
  448. }
  449. }
  450. }
  451. CSV::downloadSendHeaders('prix_produits.csv');
  452. echo CSV::array2csv($data);
  453. die();
  454. }
  455. /**
  456. * Recherche un produit en fonction de son ID.
  457. */
  458. protected function findModel(int $id)
  459. {
  460. $productModule = $this->getProductModule();
  461. if (($product = $productModule->findOneProductById($id)) !== null) {
  462. return $product;
  463. } else {
  464. throw new NotFoundHttpException('The requested page does not exist.');
  465. }
  466. }
  467. protected function findModelProductPrice($id)
  468. {
  469. $productPriceModule = $this->getProductPriceModule();
  470. if (($productPrice = $productPriceModule->findOneProductPriceById($id)) !== null) {
  471. return $productPrice;
  472. } else {
  473. throw new NotFoundHttpException('The requested page does not exist.');
  474. }
  475. }
  476. }