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.

594 líneas
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 backend\models\CreditForm;
  39. use common\helpers\GlobalParam;
  40. use common\helpers\Mailjet;
  41. use backend\models\MailForm;
  42. use common\helpers\Password;
  43. use common\logic\Distribution\Distribution\Distribution;
  44. use common\logic\Order\Order\OrderSearch;
  45. use common\logic\PointSale\PointSale\PointSale;
  46. use common\logic\PointSale\UserPointSale\UserPointSale;
  47. use common\logic\Producer\Producer\Producer;
  48. use common\logic\User\CreditHistory\CreditHistory;
  49. use common\logic\User\User\UserSearch;
  50. use common\logic\User\UserGroup\UserGroup;
  51. use common\logic\User\UserGroup\UserUserGroup;
  52. use common\logic\User\UserProducer\UserProducer;
  53. use yii\base\UserException;
  54. use yii\debug\models\search\User;
  55. use yii\filters\AccessControl;
  56. use yii\filters\VerbFilter;
  57. use \Yii;
  58. use yii\web\NotFoundHttpException;
  59. /**
  60. * UserController implements the CRUD actions for User model.
  61. */
  62. class UserController extends BackendController
  63. {
  64. public function behaviors()
  65. {
  66. return [
  67. 'verbs' => [
  68. 'class' => VerbFilter::class,
  69. 'actions' => [
  70. ],
  71. ],
  72. 'access' => [
  73. 'class' => AccessControl::class,
  74. 'rules' => [
  75. [
  76. 'allow' => true,
  77. 'roles' => ['@'],
  78. 'matchCallback' => function ($rule, $action) {
  79. return $this->isUserCurrentAdmin();
  80. }
  81. ]
  82. ],
  83. ],
  84. ];
  85. }
  86. /**
  87. * Liste les utilisateurs.
  88. *
  89. * @return mixed
  90. */
  91. public function actionIndex(
  92. $idPointSale = 0, $sectionSubscribers = false, $sectionInactiveUsers = false)
  93. {
  94. $searchModel = new UserSearch();
  95. $dataProvider = $searchModel->search([
  96. 'UserSearch' => array_merge(
  97. [
  98. 'id_point_sale' => $idPointSale,
  99. 'inactive' => (int)$sectionInactiveUsers,
  100. 'subscribers' => (int)$sectionSubscribers
  101. ],
  102. isset(Yii::$app->request->queryParams['UserSearch']) ?
  103. Yii::$app->request->queryParams['UserSearch'] :
  104. []
  105. )
  106. ]);
  107. $producer = Producer::searchOne([
  108. 'id' => GlobalParam::getCurrentProducerId()
  109. ]);
  110. $pointsSaleArray = PointSale::searchAll();
  111. return $this->render('index', [
  112. 'searchModel' => $searchModel,
  113. 'dataProvider' => $dataProvider,
  114. 'producer' => $producer,
  115. 'idPointSaleActive' => $idPointSale,
  116. 'pointsSaleArray' => $pointsSaleArray,
  117. 'sectionInactiveUsers' => $sectionInactiveUsers,
  118. 'sectionSubscribers' => $sectionSubscribers,
  119. ]);
  120. }
  121. public function initForm($model)
  122. {
  123. if ($model->id) {
  124. // init points de vente sélectionnés
  125. $userPointSaleArray = UserPointSale::searchAll([
  126. 'id_user' => $model->id
  127. ]);
  128. if ($userPointSaleArray && count($userPointSaleArray) > 0) {
  129. foreach ($userPointSaleArray as $userPointSaleArray) {
  130. $model->points_sale[] = $userPointSaleArray->id_point_sale;
  131. }
  132. }
  133. // init groupes d'utilisateurs sélectionnés
  134. $userUserGroupsArray = UserUserGroup::searchAll([
  135. 'id_user' => $model->id
  136. ]);
  137. if ($userUserGroupsArray && count($userUserGroupsArray) > 0) {
  138. foreach ($userUserGroupsArray as $userUserGroup) {
  139. $model->user_groups[] = $userUserGroup->id_user_group;
  140. }
  141. }
  142. // product price percent
  143. $userProducer = UserProducer::searchOne([
  144. 'id_producer' => GlobalParam::getCurrentProducerId(),
  145. 'id_user' => $model->id
  146. ]);
  147. $model->product_price_percent = $userProducer->product_price_percent;
  148. }
  149. // points de vente
  150. $pointsSaleArray = PointSale::find()
  151. ->where([
  152. 'id_producer' => GlobalParam::getCurrentProducerId(),
  153. 'status' => 1
  154. ])
  155. ->joinWith(['userPointSale' => function ($query) use ($model) {
  156. if ($model->id) {
  157. $query->andOnCondition('user_point_sale.id_user = ' . $model->id);
  158. }
  159. }])
  160. ->all();
  161. // groupes d'utilisateurs
  162. $userGroupsArray = UserGroup::find()
  163. ->where([
  164. 'id_producer' => GlobalParam::getCurrentProducerId(),
  165. ])
  166. ->all();
  167. return [
  168. 'pointsSaleArray' => $pointsSaleArray,
  169. 'userGroupsArray' => $userGroupsArray,
  170. ];
  171. }
  172. /**
  173. * Creates a new User model.
  174. * If creation is successful, the browser will be redirected to the 'view' page.
  175. * @return mixed
  176. */
  177. public function actionCreate()
  178. {
  179. $model = new User();
  180. $userExist = false;
  181. $posts = Yii::$app->request->post();
  182. if ($posts && isset($posts['User']['email']) && strlen($posts['User']['email']) > 0) {
  183. $userExist = User::searchOne([
  184. 'email' => $posts['User']['email']
  185. ]);
  186. }
  187. if ($userExist) {
  188. Producer::addUser($userExist->id, GlobalParam::getCurrentProducerId());
  189. $this->processLinkPointSale($userExist);
  190. $this->processLinkUserGroup($userExist);
  191. Yii::$app->getSession()->setFlash('success', "L'utilisateur que vous souhaitez créer possède déjà un compte sur la plateforme. Il vient d'être lié à votre établissement.");
  192. } else {
  193. if ($model->load(Yii::$app->request->post()) && $model->validate() && YII_ENV != 'demo') {
  194. // save user
  195. $password = Password::generate();
  196. $model->id_producer = 0;
  197. $model->setPassword($password);
  198. $model->generateAuthKey();
  199. $model->username = $model->email;
  200. if (!strlen($model->email)) {
  201. $model->username = 'inconnu@opendistrib.net';
  202. }
  203. $model->save();
  204. // liaison etablissement / user
  205. $useProducer = new UserProducer();
  206. $useProducer->id_user = $model->id;
  207. $useProducer->id_producer = GlobalParam::getCurrentProducerId();
  208. $useProducer->credit = 0;
  209. $useProducer->active = 1;
  210. $useProducer->save();
  211. $model->sendMailWelcome($password);
  212. $this->processLinkPointSale($model);
  213. $this->processLinkUserGroup($model);
  214. $this->processProductPricePercent($model);
  215. Yii::$app->getSession()->setFlash('success', 'Utilisateur créé.');
  216. $model = new User();
  217. }
  218. }
  219. return $this->render('create', array_merge($this->initForm($model), [
  220. 'model' => $model,
  221. ]));
  222. }
  223. /**
  224. * Updates an existing User model.
  225. * If update is successful, the browser will be redirected to the 'view' page.
  226. * @param integer $id
  227. * @return mixed
  228. */
  229. public function actionUpdate($id)
  230. {
  231. $model = $this->findModel($id);
  232. // Moodification du profil
  233. $previousMail = $model->email;
  234. $user = User::find()->with('userProducer')->where(['id' => $model['id']])->one();
  235. $userBelongToProducer = UserProducer::findOne(['id_user' => $id, 'id_producer' => GlobalParam::getCurrentProducerId()]);
  236. if ($userBelongToProducer) {
  237. if ($model->load(Yii::$app->request->post()) && $model->save()) {
  238. // on envoie le mail de bienvenue si le mail vient d'être défini
  239. if (!strlen($previousMail) && strlen($model->email)) {
  240. $password = Password::generate();
  241. $model->setPassword($password);
  242. $model->username = $model->email;
  243. $model->sendMailWelcome($password);
  244. }
  245. $this->processLinkPointSale($model);
  246. $this->processLinkUserGroup($model);
  247. $this->processProductPricePercent($model);
  248. Yii::$app->getSession()->setFlash('success', 'Utilisateur modifié.');
  249. }
  250. } else {
  251. throw new UserException("Vous ne pouvez pas modifier cet utilisateur.");
  252. }
  253. // Nouveau mot de passe
  254. $newPassword = Yii::$app->request->post('submit_new_password');
  255. if ($newPassword) {
  256. $password = Password::generate();
  257. $model->setPassword($password);
  258. $model->save();
  259. $producer = GlobalParam::getCurrentProducer();
  260. Mailjet::sendMail([
  261. 'from_email' => $producer->getEmailOpendistrib(),
  262. 'from_name' => $producer->name,
  263. 'to_email' => $model->email,
  264. 'to_name' => $model->getUsername(),
  265. 'subject' => '[' . $producer->name . '] Nouveau mot de passe',
  266. 'content_view_text' => '@common/mail/newPasswordUserAdmin-text.php',
  267. 'content_view_html' => '@common/mail/newPasswordUserAdmin-html.php',
  268. 'content_params' => [
  269. 'user' => $user,
  270. 'producer' => $producer,
  271. 'password' => $password,
  272. ]
  273. ]);
  274. Yii::$app->getSession()->setFlash('success', 'Nouveau mot de passe envoyé.');
  275. }
  276. return $this->render('update', array_merge($this->initForm($model), [
  277. 'model' => $model,
  278. ]));
  279. }
  280. /**
  281. * Lie un utilisateur aux points de vente sélectionnés.
  282. *
  283. * @param User $modelUser
  284. */
  285. public function processLinkPointSale($modelUser)
  286. {
  287. $posts = Yii::$app->request->post();
  288. UserPointSale::deleteAll([
  289. 'id_user' => $modelUser->id
  290. ]);
  291. if (is_array($modelUser->points_sale) && count($modelUser->points_sale) > 0) {
  292. foreach ($modelUser->points_sale as $pointSaleId) {
  293. $userPointSale = UserPointSale::searchOne([
  294. 'id_user' => $modelUser->id,
  295. 'id_point_sale' => $pointSaleId
  296. ]);
  297. if (!$userPointSale) {
  298. $userPointSale = new UserPointSale;
  299. $userPointSale->id_user = $modelUser->id;
  300. $userPointSale->id_point_sale = $pointSaleId;
  301. $userPointSale->comment = isset($posts['User']['comment_point_sale_' . $pointSaleId]) ? $posts['User']['comment_point_sale_' . $pointSaleId] : '';
  302. $userPointSale->save();
  303. }
  304. }
  305. }
  306. }
  307. /**
  308. * Lie un utilisateur aux groupes d'utilisateurs sélectionnés.
  309. *
  310. * @param User $modelUser
  311. */
  312. public function processLinkUserGroup($modelUser)
  313. {
  314. $posts = Yii::$app->request->post();
  315. UserUserGroup::deleteAll([
  316. 'id_user' => $modelUser->id
  317. ]);
  318. if (is_array($modelUser->user_groups) && count($modelUser->user_groups) > 0) {
  319. foreach ($modelUser->user_groups as $userGroupId) {
  320. $userUserGroup = UserUserGroup::searchOne([
  321. 'id_user' => $modelUser->id,
  322. 'id_user_group' => $userGroupId
  323. ]);
  324. if (!$userUserGroup) {
  325. $userUserGroup = new UserUserGroup();
  326. $userUserGroup->id_user = $modelUser->id;
  327. $userUserGroup->id_user_group = $userGroupId;
  328. $userUserGroup->save();
  329. }
  330. }
  331. }
  332. }
  333. public function processProductPricePercent($model)
  334. {
  335. $userProducer = UserProducer::searchOne([
  336. 'id_producer' => GlobalParam::getCurrentProducerId(),
  337. 'id_user' => $model->id
  338. ]);
  339. $userProducer->product_price_percent = $model->product_price_percent;
  340. $userProducer->save();
  341. }
  342. /**
  343. * Désactive l'utilisateur de l'établissement.
  344. *
  345. * @param integer $id ID de l'utilisateur
  346. */
  347. public function actionDelete($id)
  348. {
  349. $userProducer = UserProducer::findOne([
  350. 'id_user' => $id,
  351. 'id_producer' => GlobalParam::getCurrentProducerId()
  352. ]);
  353. if ($userProducer) {
  354. $userProducer->active = 0;
  355. $userProducer->bookmark = 0;
  356. $userProducer->save();
  357. Yii::$app->getSession()->setFlash('success', 'L\'utilisateur a bien été supprimé de votre établissement.');
  358. } else {
  359. throw new \yii\web\NotFoundHttpException('L\'enregistrement UserProducer est introuvable', 404);
  360. }
  361. $params = Yii::$app->getRequest()->getQueryParams();
  362. unset($params['id']);
  363. $this->redirect(array_merge(['index'], $params));
  364. }
  365. /**
  366. * Affiche la liste des emails des utilisateurs liés à un point de vente
  367. * donné.
  368. *
  369. * @param integer $idPointSale
  370. * @return mixed
  371. */
  372. public function actionMail(
  373. $idPointSale = 0,
  374. $sectionSubscribers = 0,
  375. $sectionInactiveUsers = 0,
  376. $usersPointSaleLink = 0,
  377. $usersPointSaleHasOrder = 0)
  378. {
  379. if ($idPointSale && !$usersPointSaleLink && !$usersPointSaleHasOrder) {
  380. $usersPointSaleLink = 1;
  381. }
  382. $users = User::findBy([
  383. 'id_producer' => GlobalParam::getCurrentProducerId(),
  384. 'id_point_sale' => $idPointSale,
  385. 'users_point_sale_link' => $usersPointSaleLink,
  386. 'users_point_sale_has_order' => $usersPointSaleHasOrder,
  387. 'subscribers' => $sectionSubscribers,
  388. 'inactive' => $sectionInactiveUsers,
  389. ])->all();
  390. $usersArray = [];
  391. foreach ($users as $user) {
  392. if (isset($user['email']) && strlen($user['email']))
  393. $usersArray[] = $user['email'];
  394. }
  395. $pointsSaleArray = PointSale::find()->where(['id_producer' => GlobalParam::getCurrentProducerId()])->all();
  396. $pointSale = null;
  397. if ($idPointSale) {
  398. $pointSale = PointSale::findOne(['id' => $idPointSale]);
  399. }
  400. $mailForm = new MailForm();
  401. if ($mailForm->load(Yii::$app->request->post()) && $mailForm->validate()) {
  402. $responseSendMail = $mailForm->sendEmail($users);
  403. if ($responseSendMail->success()) {
  404. Yii::$app->getSession()->setFlash('success', 'Votre email a bien été envoyé.');
  405. } else {
  406. $bodyResponseSendMail = $responseSendMail->getBody();
  407. $emailsErrorArray = [];
  408. if(isset($bodyResponseSendMail['Messages'])) {
  409. foreach ($bodyResponseSendMail['Messages'] as $message) {
  410. if ($message['Status'] != 'success') {
  411. $emailsErrorArray[] = $message['To']['Email'];
  412. }
  413. }
  414. }
  415. $messageError = 'Un problème est survenu lors de l\'envoi de votre email.';
  416. if (count($emailsErrorArray) > 0) {
  417. $messageError .= '<br />Problème détecté sur les adresses suivantes : ' . implode(',', $emailsErrorArray);
  418. }
  419. Yii::$app->getSession()->setFlash('error', $messageError);
  420. }
  421. return $this->redirect(['mail', 'idPointSale' => $idPointSale]);
  422. }
  423. $incomingDistributions = Distribution::getIncoming();
  424. $incomingDistributionsArray = ['0' => '--'];
  425. foreach ($incomingDistributions as $distribution) {
  426. $incomingDistributionsArray[$distribution->id] = strftime('%A %d %B %Y', strtotime($distribution->date));
  427. }
  428. return $this->render('emails', [
  429. 'usersArray' => $usersArray,
  430. 'pointsSaleArray' => $pointsSaleArray,
  431. 'pointSale' => $pointSale,
  432. 'mailForm' => $mailForm,
  433. 'idPointSaleActive' => $idPointSale,
  434. 'incomingDistributionsArray' => $incomingDistributionsArray,
  435. 'sectionSubscribers' => $sectionSubscribers,
  436. 'sectionInactiveUsers' => $sectionInactiveUsers,
  437. 'usersPointSaleLink' => $usersPointSaleLink,
  438. 'usersPointSaleHasOrder' => $usersPointSaleHasOrder,
  439. ]);
  440. }
  441. /**
  442. * Affiche les données liées au crédit d'un utilisateur (formulaire,
  443. * historique).
  444. *
  445. * @param integer $id
  446. * @return mixed
  447. * @throws UserException
  448. */
  449. public function actionCredit($id)
  450. {
  451. $user = User::find()->with('userProducer')->where(['id' => $id])->one();
  452. $userProducer = UserProducer::findOne(['id_user' => $id, 'id_producer' => GlobalParam::getCurrentProducerId()]);
  453. if (($userProducer) || User::getCurrentStatus() == User::STATUS_ADMIN) {
  454. $creditForm = new CreditForm();
  455. if ($creditForm->load(Yii::$app->request->post()) && $creditForm->validate()) {
  456. $creditForm->id_user = $id;
  457. $creditForm->save();
  458. $creditForm = new CreditForm;
  459. }
  460. $history = CreditHistory::find()
  461. ->with(['order', 'userAction'])
  462. ->where([
  463. 'id_user' => $user->id,
  464. 'id_producer' => GlobalParam::getCurrentProducerId(),
  465. ])
  466. ->orderBy('date DESC')
  467. ->all();
  468. return $this->render('credit', [
  469. 'user' => $user,
  470. 'userProducer' => $userProducer,
  471. 'creditForm' => $creditForm,
  472. 'history' => $history
  473. ]);
  474. } else {
  475. throw new UserException("Vous ne pouvez pas créditer un utilisateur qui n'est pas associé à votre établissement.");
  476. }
  477. }
  478. /**
  479. * Affiche les commandes d'un utilisateur.
  480. *
  481. * @param integer $id
  482. * @return mixed
  483. */
  484. public function actionOrders($id)
  485. {
  486. $user = User::findOne($id);
  487. $searchModel = new OrderSearch();
  488. $dataProvider = $searchModel->search(array_merge(Yii::$app->request->queryParams, ['id_user' => $id]));
  489. return $this->render('orders', [
  490. 'user' => $user,
  491. 'searchModel' => $searchModel,
  492. 'dataProvider' => $dataProvider,
  493. ]);
  494. }
  495. /**
  496. * Modifie l'option "credit_active" d'un utilisateur pour le producteur courant.
  497. * Redirige vers la page de crédit de l'utilisateur.
  498. *
  499. * @param integer $idUser
  500. * @param boolean $state
  501. */
  502. public function actionStateCredit($idUser, $state)
  503. {
  504. $userProducer = UserProducer::searchOne([
  505. 'id_user' => $idUser
  506. ]);
  507. if ($userProducer) {
  508. $userProducer->credit_active = $state;
  509. $userProducer->save();
  510. }
  511. return $this->redirect(['user/credit', 'id' => $idUser]);
  512. }
  513. /**
  514. * Finds the User model based on its primary key value.
  515. * If the model is not found, a 404 HTTP exception will be thrown.
  516. */
  517. protected function findModel($id)
  518. {
  519. if (($model = User::findOne($id)) !== null) {
  520. return $model;
  521. } else {
  522. throw new NotFoundHttpException('The requested page does not exist.');
  523. }
  524. }
  525. }