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

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