Vous ne pouvez pas sélectionner plus de 25 sujets Les noms de sujets doivent commencer par une lettre ou un nombre, peuvent contenir des tirets ('-') et peuvent comporter jusqu'à 35 caractères.

605 lines
23KB

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