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.

609 líneas
27KB

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