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.

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