573 lines
24KB

  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\models\DeliveryNote;
  39. use common\models\Product;
  40. use common\models\User;
  41. use common\models\Document;
  42. use common\helpers\GlobalParam;
  43. use common\models\Order;
  44. use yii\base\UserException;
  45. use yii;
  46. class DocumentController extends BackendController
  47. {
  48. public function behaviors()
  49. {
  50. return [
  51. 'verbs' => [
  52. 'class' => VerbFilter::className(),
  53. 'actions' => [
  54. ],
  55. ],
  56. 'access' => [
  57. 'class' => AccessControl::className(),
  58. 'rules' => [
  59. [
  60. 'allow' => true,
  61. 'roles' => ['@'],
  62. 'matchCallback' => function ($rule, $action) {
  63. return User::hasAccessBackend();
  64. }
  65. ]
  66. ],
  67. ],
  68. ];
  69. }
  70. public function actionCreate()
  71. {
  72. $class = $this->getClass();
  73. $model = new $class();
  74. if ($model->load(Yii::$app->request->post())) {
  75. $model->id_producer = GlobalParam::getCurrentProducerId();
  76. if ($model->save()) {
  77. $this->processInvoiceViaDeliveryNotes($model) ;
  78. Yii::$app->getSession()->setFlash('success', $this->getFlashMessage('create', $model));
  79. return $this->redirect(['/' . $this->getControllerUrl() . '/update', 'id' => $model->id]);
  80. } else {
  81. Yii::$app->getSession()->setFlash('error', 'Un problème est survenu lors de la création du document.');
  82. }
  83. }
  84. return $this->render('/document/create', [
  85. 'title' => $this->getTitle('Ajouter'),
  86. 'typeDocument' => $this->getDocumentType(),
  87. 'model' => $model,
  88. ]);
  89. }
  90. public function processInvoiceViaDeliveryNotes($model)
  91. {
  92. if($model->getClass() == 'Invoice') {
  93. if($model->deliveryNotes && is_array($model->deliveryNotes) && count($model->deliveryNotes)) {
  94. foreach($model->deliveryNotes as $key => $idDeliveryNote) {
  95. Order::updateAll([
  96. 'id_invoice' => $model->id
  97. ], [
  98. 'id_delivery_note' => $idDeliveryNote
  99. ]) ;
  100. }
  101. }
  102. }
  103. }
  104. /**
  105. * Modifie un modèle Produit existant.
  106. * Si la modification réussit, le navigateur est redirigé vers la page 'index'.
  107. *
  108. * @param integer $id
  109. * @return mixed
  110. */
  111. public function actionUpdate($id)
  112. {
  113. $model = $this->findModel($id);
  114. if(!$model) {
  115. throw new NotFoundHttpException('Le document n\'a pas été trouvé.');
  116. }
  117. if ($model->isStatusValid()) {
  118. return $this->redirect(['download', 'id' => $id]) ;
  119. }
  120. if ($model && $model->load(Yii::$app->request->post()) && $model->save()) {
  121. Yii::$app->getSession()->setFlash('success', $this->getFlashMessage('update', $model));
  122. }
  123. return $this->render('/document/update', [
  124. 'title' => $this->getTitle('Modifier'),
  125. 'typeDocument' => $this->getDocumentType(),
  126. 'model' => $model,
  127. ]);
  128. }
  129. public function actionDelete($id)
  130. {
  131. $model = $this->findModel($id);
  132. if ($model->isStatusValid()) {
  133. throw new UserException('Vous ne pouvez pas supprimer un document validé.');
  134. }
  135. $model->delete();
  136. if($this->getClass() == 'DeliveryNote') {
  137. Order::updateAll([
  138. 'order.id_delivery_note' => null
  139. ], [
  140. 'order.id_delivery_note' => $id
  141. ]);
  142. }
  143. if($this->getClass() == 'Quotation') {
  144. Order::updateAll([
  145. 'order.id_quotation' => null
  146. ], [
  147. 'order.id_quotation' => $id
  148. ]);
  149. }
  150. if($this->getClass() == 'Invoice') {
  151. Order::updateAll([
  152. 'order.id_invoice' => null
  153. ], [
  154. 'order.id_invoice' => $id
  155. ]);
  156. }
  157. Yii::$app->getSession()->setFlash('success', $this->getFlashMessage('delete', $model));
  158. $this->redirect([$this->getControllerUrl().'/index']);
  159. }
  160. public function actionDownload($id)
  161. {
  162. $document = $this->findModel($id);
  163. return $document->generatePdf(Pdf::DEST_BROWSER) ;
  164. }
  165. public function actionSend($id)
  166. {
  167. $document = $this->findModel($id) ;
  168. if($document->send()) {
  169. Yii::$app->getSession()->setFlash('success', $this->getFlashMessage('send', $document));
  170. }
  171. else {
  172. Yii::$app->getSession()->setFlash('danger', $this->getFlashMessage('send', $document));
  173. }
  174. $this->redirect([$this->getControllerUrl().'/index']);
  175. }
  176. public function actionAjaxUserInfos($typeAction, $idUser, $classDocument, $idDocument = false)
  177. {
  178. \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
  179. if ($idUser > 0) {
  180. $user = User::searchOne([
  181. 'id' => $idUser
  182. ]);
  183. if ($user) {
  184. $document = null ;
  185. if(Document::isValidClass($classDocument)) {
  186. $document = $classDocument::searchOne([
  187. 'id' => $idDocument,
  188. 'id_user' => $idUser
  189. ]) ;
  190. }
  191. if($document && $document->id_user == $user->id) {
  192. $address = $document->address ;
  193. }
  194. else {
  195. $address = $user->getFullAddress() ;
  196. }
  197. $json = [
  198. 'return' => 'success',
  199. 'address' => $address
  200. ];
  201. if($classDocument == 'Invoice') {
  202. if($typeAction == 'create') {
  203. $deliveryNotesArray = DeliveryNote::searchAll([
  204. 'id_user' => $user->id,
  205. 'status' => Document::STATUS_VALID
  206. ]) ;
  207. }
  208. elseif($typeAction == 'update' && $idDocument > 0) {
  209. $deliveryNotesArray = DeliveryNote::searchAll([
  210. 'id_user' => $user->id,
  211. 'status' => Document::STATUS_VALID,
  212. 'order.id_invoice' => $idDocument
  213. ]) ;
  214. }
  215. if(isset($deliveryNotesArray)) {
  216. $json['delivery_notes'] = [] ;
  217. foreach($deliveryNotesArray as $deliveryNote) {
  218. $json['delivery_notes'][] = array_merge(
  219. $deliveryNote->getAttributes(),
  220. [
  221. 'total' => $deliveryNote->getAmountWithTax()
  222. ]
  223. ) ;
  224. }
  225. }
  226. }
  227. return $json ;
  228. }
  229. }
  230. return ['return' => 'error'];
  231. }
  232. public function actionValidate($id)
  233. {
  234. $classDocument = $this->getClass();
  235. if ($id > 0 && Document::isValidClass($classDocument)) {
  236. $document = $classDocument::searchOne([
  237. 'id' => $id
  238. ]);
  239. if ($document) {
  240. $document->changeStatus(Document::STATUS_VALID);
  241. $document->save();
  242. Yii::$app->getSession()->setFlash('success', $this->getFlashMessage('validate', $document));
  243. return $this->redirect([$this->getControllerUrl().'/index']);
  244. }
  245. }
  246. Yii::$app->getSession()->setFlash('danger', 'Une erreur est survenue lors de la validation du document.');
  247. return $this->redirect([$this->getControllerUrl().'/index']);
  248. }
  249. public function actionAjaxValidateDocument($idDocument, $classDocument)
  250. {
  251. \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
  252. if ($idDocument > 0 && Document::isValidClass($classDocument)) {
  253. $document = $classDocument::searchOne([
  254. 'id' => $idDocument
  255. ]);
  256. if ($document) {
  257. $document->changeStatus(Document::STATUS_VALID);
  258. $document->save();
  259. return [
  260. 'return' => 'success',
  261. 'alert' => [
  262. 'type' => 'success',
  263. 'message' => 'Document validé'
  264. ]
  265. ];
  266. }
  267. }
  268. return [
  269. 'return' => 'error',
  270. 'alert' => [
  271. 'type' => 'danger',
  272. 'message' => 'Une erreur est survenue lors de la validation du document.'
  273. ]
  274. ];
  275. }
  276. public function actionAjaxInit($idDocument, $classDocument)
  277. {
  278. \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
  279. if ($idDocument > 0 && Document::isValidClass($classDocument)) {
  280. $document = $classDocument::searchOne([
  281. 'id' => $idDocument
  282. ]);
  283. if ($document) {
  284. $productsArray = Product::searchAll();
  285. $ordersArray = [];
  286. foreach ($document->orders as $order) {
  287. $order->init();
  288. $productsOrderArray = [];
  289. foreach ($order->productOrder as $productOrder) {
  290. $productsOrderArray[$productOrder->id] = $productOrder->getAttributes();
  291. }
  292. $ordersArray[$order->id] = array_merge(
  293. $order->getAttributes(),
  294. [
  295. 'username' => $order->getUsername(),
  296. 'distribution_date' => isset($order->distribution) ? date('d/m/Y', strtotime($order->distribution->date)) : null,
  297. 'point_sale_name' => isset($order->pointSale) ? $order->pointSale->name : null,
  298. 'productOrder' => $productsOrderArray,
  299. ]
  300. );
  301. }
  302. return [
  303. 'return' => 'success',
  304. 'tax_rate_producer' => GlobalParam::getCurrentProducer()->taxRate->value,
  305. 'document' => array_merge($document->getAttributes(), [
  306. 'html_label' => $document->getHtmlLabel(),
  307. 'class' => $document->getClass()
  308. ]),
  309. 'id_user' => $document->user->id,
  310. 'products' => ArrayHelper::map($productsArray, 'id', function ($product) {
  311. return array_merge($product->getAttributes(), [
  312. 'price_with_tax' => $product->price_with_tax,
  313. 'wording_unit' => $product->wording_unit,
  314. 'tax_rate' => $product->taxRate->value
  315. ]);
  316. }),
  317. 'orders' => $ordersArray,
  318. 'total' => $document->getAmount(Order::AMOUNT_TOTAL),
  319. 'total_with_tax' => $document->getAmountWithTax(Order::AMOUNT_TOTAL),
  320. ];
  321. }
  322. }
  323. return ['return' => 'error'];
  324. }
  325. public function actionAjaxAddProduct($idDocument, $classDocument, $idProduct, $quantity, $price)
  326. {
  327. \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
  328. if (Document::isValidClass($classDocument)) {
  329. $document = $classDocument::searchOne([
  330. 'id' => $idDocument
  331. ]);
  332. $product = Product::searchOne([
  333. 'id' => $idProduct
  334. ]);
  335. if ($document && $product) {
  336. if (count($document->orders) == 0) {
  337. $order = new Order;
  338. $order->id_user = $document->id_user;
  339. $order->id_point_sale = null;
  340. $order->id_distribution = null;
  341. $order->status = 'tmp-order' ;
  342. $order->origin = Order::ORIGIN_ADMIN;
  343. $order->date = date('Y-m-d H:i:s');
  344. $fieldIdDocument = 'id_' . $classDocument::tableName();
  345. $order->$fieldIdDocument = $document->id;
  346. $order->save();
  347. } else {
  348. $order = $document->orders[0];
  349. }
  350. if ($order) {
  351. $productOrder = new ProductOrder;
  352. $productOrder->id_order = $order->id;
  353. $productOrder->id_product = $idProduct;
  354. $quantity = $quantity / Product::$unitsArray[$product->unit]['coefficient'];
  355. $productOrder->quantity = $quantity;
  356. $productOrder->price = (float)$price;
  357. $productOrder->unit = $product->unit;
  358. $productOrder->step = $product->step;
  359. $productOrder->id_tax_rate = $product->taxRate->id ;
  360. $productOrder->save();
  361. return [
  362. 'return' => 'success',
  363. 'alert' => [
  364. 'type' => 'success',
  365. 'message' => 'Produit ajouté'
  366. ]
  367. ];
  368. }
  369. }
  370. }
  371. return [
  372. 'return' => 'error',
  373. 'alert' => [
  374. 'type' => 'danger',
  375. 'message' => 'Une erreur est survenue lors de la suppression du produit.'
  376. ]
  377. ];
  378. }
  379. public function actionAjaxDeleteProductOrder($idProductOrder)
  380. {
  381. \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
  382. $productOrder = ProductOrder::searchOne([
  383. 'id' => $idProductOrder
  384. ]);
  385. if ($productOrder) {
  386. $productOrder->delete();
  387. return [
  388. 'return' => 'success',
  389. 'alert' => [
  390. 'type' => 'danger',
  391. 'message' => 'Produit supprimé'
  392. ]
  393. ];
  394. }
  395. return [
  396. 'return' => 'error',
  397. 'alert' => [
  398. 'type' => 'danger',
  399. 'message' => 'Une erreur est survenue lors de la suppression du produit.'
  400. ]
  401. ];
  402. }
  403. public function getClass()
  404. {
  405. $class = get_class($this);
  406. $class = str_replace('Controller', '', $class);
  407. $class = str_replace('backend\controllers\\', '', $class);
  408. return $class;
  409. }
  410. public function getDocumentType()
  411. {
  412. $class = $this->getClass();
  413. if ($class == 'Invoice') {
  414. $documentType = 'Facture';
  415. } elseif ($class == 'DeliveryNote') {
  416. $documentType = 'Bon de livraison';
  417. } elseif ($class == 'Quotation') {
  418. $documentType = 'Devis';
  419. }
  420. if (isset($documentType)) {
  421. return $documentType;
  422. }
  423. return '';
  424. }
  425. public function getFlashMessage($type = 'create', $model)
  426. {
  427. $class = $this->getClass();
  428. $message = $this->getDocumentType();
  429. $message .= ' <strong>' . Html::encode($model->name) . '</strong> ';
  430. if ($type == 'create') {
  431. $message .= 'ajouté';
  432. } elseif ($type == 'update') {
  433. $message .= 'modifié';
  434. } elseif ($type == 'delete') {
  435. $message .= 'supprimé';
  436. } elseif ($type == 'validate') {
  437. $message .= 'validé';
  438. } elseif ($type == 'send') {
  439. $message .= 'envoyé';
  440. }
  441. if ($class == 'Invoice') {
  442. $message .= 'e';
  443. }
  444. return $message;
  445. }
  446. protected function getTitle($prepend)
  447. {
  448. $class = $this->getClass();
  449. switch ($class) {
  450. case 'Invoice' :
  451. $title = $prepend . ' une facture';
  452. break;
  453. case 'DeliveryNote' :
  454. $title = $prepend . ' un bon de livraison';
  455. break;
  456. case 'Quotation' :
  457. $title = $prepend . ' un devis';
  458. break;
  459. }
  460. return $title;
  461. }
  462. public function getControllerUrl()
  463. {
  464. $path = strtolower($this->getClass());
  465. $path = str_replace('deliverynote', 'delivery-note', $path);
  466. return $path;
  467. }
  468. /**
  469. * Recherche un Document en fonction de son ID.
  470. *
  471. * @param integer $id
  472. * @return Document
  473. * @throws NotFoundHttpException si le modèle n'est pas trouvé
  474. */
  475. protected function findModel($id)
  476. {
  477. $class = $this->getClass();
  478. $model = $class::searchOne([
  479. 'id' => $id
  480. ]);
  481. if ($model) {
  482. return $model;
  483. } else {
  484. throw new NotFoundHttpException('The requested page does not exist.');
  485. }
  486. }
  487. }