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.

597 line
22KB

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