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.

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