Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

8 роки тому
6 роки тому
8 роки тому
8 роки тому
8 роки тому
8 роки тому
8 роки тому
8 роки тому
8 роки тому
8 роки тому
8 роки тому
8 роки тому
8 роки тому
8 роки тому
8 роки тому
8 роки тому
8 роки тому
8 роки тому
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388
  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 common\helpers\GlobalParam;
  39. use common\models\ProductDistribution;
  40. use common\models\ProductPrice;
  41. use common\models\ProductPriceSearch;
  42. use common\models\ProductSearch;
  43. use common\models\UserSearch;
  44. use Yii;
  45. use yii\filters\AccessControl;
  46. use common\models\Product;
  47. use common\models\Distribution;
  48. use common\models\User;
  49. use common\models\UserProducer;
  50. use yii\data\ActiveDataProvider;
  51. use yii\web\Controller;
  52. use yii\web\NotFoundHttpException;
  53. use yii\filters\VerbFilter;
  54. use yii\web\UploadedFile;
  55. use common\helpers\Upload;
  56. /**
  57. * ProduitController implements the CRUD actions for Produit model.
  58. */
  59. class ProductController extends BackendController
  60. {
  61. var $enableCsrfValidation = false;
  62. public function behaviors()
  63. {
  64. return [
  65. 'verbs' => [
  66. 'class' => VerbFilter::className(),
  67. 'actions' => [
  68. ],
  69. ],
  70. 'access' => [
  71. 'class' => AccessControl::className(),
  72. 'rules' => [
  73. [
  74. 'allow' => true,
  75. 'roles' => ['@'],
  76. 'matchCallback' => function ($rule, $action) {
  77. return User::hasAccessBackend();
  78. }
  79. ]
  80. ],
  81. ],
  82. ];
  83. }
  84. /**
  85. * Liste les modèles Produit.
  86. *
  87. * @return mixed
  88. */
  89. public function actionIndex()
  90. {
  91. $searchModel = new ProductSearch();
  92. $dataProvider = $searchModel->search(Yii::$app->request->queryParams);
  93. return $this->render('index', [
  94. 'searchModel' => $searchModel,
  95. 'dataProvider' => $dataProvider,
  96. ]);
  97. }
  98. /**
  99. * Crée un modèle Produit.
  100. * Si la création réussit, le navigateur est redirigé vers la page 'index'.
  101. *
  102. * @return mixed
  103. */
  104. public function actionCreate()
  105. {
  106. $model = new Product();
  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()) && $model->save()) {
  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. $model->save();
  124. // availability on points sale
  125. $this->processAvailabilityPointsSale($model);
  126. // link product / distribution
  127. Distribution::linkProductIncomingDistributions($model);
  128. Yii::$app->getSession()->setFlash('success', 'Produit <strong>' . Html::encode($model->name) . '</strong> ajouté');
  129. return $this->redirect(['index']);
  130. } else {
  131. return $this->render('create', [
  132. 'model' => $model,
  133. ]);
  134. }
  135. }
  136. /**
  137. * Modifie un modèle Produit existant.
  138. * Si la modification réussit, le navigateur est redirigé vers la page 'index'.
  139. *
  140. * @param integer $id
  141. * @return mixed
  142. */
  143. public function actionUpdate($id)
  144. {
  145. $request = Yii::$app->request;
  146. $model = $this->findModel($id);
  147. foreach ($model->productPointSale as $productPointSale) {
  148. $model->pointsSale[] = $productPointSale->id_point_sale;
  149. }
  150. $photoFilenameOld = $model->photo;
  151. if ($model->load(Yii::$app->request->post()) && $model->save()) {
  152. Upload::uploadFile($model, 'photo', $photoFilenameOld);
  153. $deletePhoto = $request->post('delete_photo', 0);
  154. if ($deletePhoto) {
  155. $model->photo = '';
  156. $model->save();
  157. }
  158. // availability on points sale
  159. $this->processAvailabilityPointsSale($model);
  160. if ($model->apply_distributions) {
  161. // link product / distribution
  162. Distribution::linkProductIncomingDistributions($model);
  163. }
  164. Yii::$app->getSession()->setFlash('success', 'Produit <strong>' . Html::encode($model->name) . '</strong> modifié');
  165. return $this->redirect(['index']);
  166. }
  167. return $this->render('update/update', [
  168. 'model' => $model,
  169. 'action' => 'update',
  170. ]);
  171. }
  172. /**
  173. * Traite les accès restreints d'un point de vente.
  174. */
  175. public function processAvailabilityPointsSale($model)
  176. {
  177. ProductPointSale::deleteAll(['id_product' => $model->id]);
  178. if (is_array($model->pointsSale) && count($model->pointsSale)) {
  179. foreach ($model->pointsSale as $key => $val) {
  180. $pointSale = PointSale::findOne($val);
  181. if ($pointSale) {
  182. $productPointSale = new ProductPointSale;
  183. $productPointSale->id_product = $model->id;
  184. $productPointSale->id_point_sale = $pointSale->id;
  185. $productPointSale->available = ($model->available_on_points_sale) ? 0 : 1;
  186. $productPointSale->save();
  187. }
  188. }
  189. }
  190. }
  191. public function actionPricesList($id)
  192. {
  193. $request = Yii::$app->request;
  194. $model = $this->findModel($id);
  195. $searchModel = new ProductPriceSearch();
  196. $searchModel->id_product = $id;
  197. $dataProvider = $searchModel->search(array_merge(Yii::$app->request->queryParams, [
  198. 'id_product' => $id
  199. ]));
  200. $userProducerWithProductPercent = UserProducer::searchAll([], [
  201. 'join_with' => ['user'],
  202. 'conditions' => 'user_producer.product_price_percent != 0',
  203. ]);
  204. $pointSaleWithProductPercent = PointSale::searchAll([], [
  205. 'conditions' => 'point_sale.product_price_percent != 0'
  206. ]);
  207. return $this->render('update/prices/list', [
  208. 'model' => $model,
  209. 'action' => 'prices-list',
  210. 'searchModel' => $searchModel,
  211. 'dataProvider' => $dataProvider,
  212. 'userProducerWithProductPercent' => $userProducerWithProductPercent,
  213. 'pointSaleWithProductPercent' => $pointSaleWithProductPercent,
  214. ]);
  215. }
  216. public function actionPricesCreate($idProduct)
  217. {
  218. $model = new ProductPrice();
  219. $model->id_product = $idProduct;
  220. $modelProduct = $this->findModel($idProduct);
  221. if ($model->load(Yii::$app->request->post())) {
  222. $conditionsProductPriceExist = [
  223. 'id_product' => $idProduct,
  224. 'id_user' => $model->id_user ? $model->id_user : null,
  225. 'id_user_group' => $model->id_user_group ? $model->id_user_group : null,
  226. 'id_point_sale' => $model->id_point_sale ? $model->id_point_sale : null,
  227. 'from_quantity' => $model->from_quantity ? $model->from_quantity : null,
  228. ];
  229. $productPriceExist = ProductPrice::findOne($conditionsProductPriceExist);
  230. if ($productPriceExist) {
  231. $productPriceExist->delete();
  232. Yii::$app->getSession()->setFlash('warning', 'Un prix existait déjà pour cet utilisateur / point de vente, il a été supprimé.');
  233. }
  234. if ($model->save()) {
  235. Yii::$app->getSession()->setFlash('success', 'Le prix a bien été ajouté.');
  236. return $this->redirect(['product/prices-list', 'id' => $idProduct]);
  237. }
  238. }
  239. return $this->render('update/prices/create', [
  240. 'model' => $model,
  241. 'modelProduct' => $modelProduct,
  242. ]);
  243. }
  244. public function actionPricesUpdate($id)
  245. {
  246. $request = Yii::$app->request;
  247. $model = $this->findModelProductPrice($id);
  248. $modelProduct = $this->findModel($model->id_product);
  249. if ($model->load(Yii::$app->request->post()) && $model->save()) {
  250. Yii::$app->getSession()->setFlash('success', 'Prix modifié');
  251. return $this->redirect(['product/prices-list', 'id' => $model->id_product]);
  252. }
  253. return $this->render('update/prices/update', [
  254. 'model' => $model,
  255. 'modelProduct' => $modelProduct,
  256. 'action' => 'prices-update',
  257. ]);
  258. }
  259. public function actionPricesDelete($id)
  260. {
  261. $productPrice = $this->findModelProductPrice($id);
  262. $productPrice->delete();
  263. Yii::$app->getSession()->setFlash('success', 'Prix supprimé');
  264. return $this->redirect(['product/prices-list', 'id' => $productPrice->id_product]);
  265. }
  266. /**
  267. * Supprime un modèle Produit.
  268. * Si la suppression réussit, le navigateur est redirigé vers la page
  269. * 'index'.
  270. *
  271. * @param integer $id
  272. * @return mixed
  273. */
  274. public function actionDelete($id, $confirm = false)
  275. {
  276. $product = $this->findModel($id);
  277. if ($confirm) {
  278. $product->delete();
  279. ProductDistribution::deleteAll(['id_product' => $id]);
  280. Yii::$app->getSession()->setFlash('success', 'Produit <strong>' . Html::encode($product->name) . '</strong> supprimé');
  281. } else {
  282. Yii::$app->getSession()->setFlash('info', 'Souhaitez-vous vraiment supprimer le produit <strong>' . Html::encode($product->name) . '</strong> ? '
  283. . Html::a('Oui', ['product/delete', 'id' => $id, 'confirm' => 1], ['class' => 'btn btn-default']) . ' ' . Html::a('Non', ['product/index'], ['class' => 'btn btn-default']));
  284. }
  285. return $this->redirect(['index']);
  286. }
  287. /**
  288. * Modifie l'ordre des produits.
  289. *
  290. * @param array $array
  291. */
  292. public function actionOrder()
  293. {
  294. $array = Yii::$app->request->post('array');
  295. $orderArray = json_decode(stripslashes($array));
  296. foreach ($orderArray as $id => $order) {
  297. $product = $this->findModel($id);
  298. $product->order = $order;
  299. $product->save();
  300. }
  301. }
  302. public function actionAjaxToggleActive($id, $active)
  303. {
  304. \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
  305. $product = $this->findModel($id);
  306. $product->active = (int) $active;
  307. return ['success', 'id' => $id, 'active' => $active];
  308. }
  309. /**
  310. * Recherche un produit en fonction de son ID.
  311. *
  312. * @param integer $id
  313. * @return Produit
  314. * @throws NotFoundHttpException si le modèle n'est pas trouvé
  315. */
  316. protected function findModel($id)
  317. {
  318. if (($model = Product::findOne($id)) !== null) {
  319. return $model;
  320. } else {
  321. throw new NotFoundHttpException('The requested page does not exist.');
  322. }
  323. }
  324. protected function findModelProductPrice($id)
  325. {
  326. if (($model = ProductPrice::findOne($id)) !== null) {
  327. return $model;
  328. } else {
  329. throw new NotFoundHttpException('The requested page does not exist.');
  330. }
  331. }
  332. }