選択できるのは25トピックまでです。 トピックは、先頭が英数字で、英数字とダッシュ('-')を使用した35文字以内のものにしてください。

522 行
17KB

  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 frontend\controllers;
  38. use common\helpers\GlobalParam;
  39. use frontend\forms\ProducerCodeForm;
  40. use GuzzleHttp\Client;
  41. use Yii;
  42. use frontend\forms\PasswordResetRequestForm;
  43. use frontend\forms\ResetPasswordForm;
  44. use frontend\forms\SignupForm;
  45. use common\forms\ContactForm;
  46. use yii\base\InvalidParamException;
  47. use yii\data\ActiveDataProvider;
  48. use yii\helpers\Html;
  49. use yii\web\BadRequestHttpException;
  50. use yii\filters\VerbFilter;
  51. use yii\filters\AccessControl;
  52. use common\forms\LoginForm;
  53. use yii\web\NotFoundHttpException;
  54. /**
  55. * Site controller
  56. */
  57. class SiteController extends FrontendController
  58. {
  59. /**
  60. * @inheritdoc
  61. */
  62. public function behaviors()
  63. {
  64. return [
  65. 'access' => [
  66. 'class' => AccessControl::class,
  67. 'only' => ['logout', 'signup'],
  68. 'rules' => [
  69. [
  70. 'actions' => ['signup'],
  71. 'allow' => true,
  72. 'roles' => ['?'],
  73. ],
  74. [
  75. 'actions' => ['logout'],
  76. 'allow' => true,
  77. 'roles' => ['@'],
  78. ],
  79. ],
  80. ],
  81. 'verbs' => [
  82. 'class' => VerbFilter::class,
  83. 'actions' => [
  84. 'logout' => ['get'],
  85. ],
  86. ],
  87. ];
  88. }
  89. /**
  90. * @inheritdoc
  91. */
  92. public function actions()
  93. {
  94. return [
  95. 'captcha' => [
  96. 'class' => 'yii\captcha\CaptchaAction',
  97. 'fixedVerifyCode' => YII_ENV_TEST ? 'testme' : null,
  98. ],
  99. ];
  100. }
  101. /**
  102. * Affiche la page d'erreur.
  103. *
  104. * @return mixed
  105. */
  106. public function actionError()
  107. {
  108. $exception = \Yii::$app->errorHandler->exception;
  109. if ($exception->getMessage() == 'Producteur introuvable'
  110. || \Yii::$app->getRequest()->getQueryParam('producer_not_found')) {
  111. return $this->render('error-404-producer', ['exception' => $exception]);
  112. }
  113. if ($exception !== null) {
  114. return $this->render('error', ['exception' => $exception]);
  115. }
  116. }
  117. /**
  118. * Affiche la page d'accueil.
  119. */
  120. public function actionIndex()
  121. {
  122. return $this->render('index');
  123. }
  124. public function actionService()
  125. {
  126. return $this->render('service', [
  127. 'producerDemoAccount' => $this->getProducerManager()->findOneProducerDemoAccount(),
  128. 'dataProviderPrices' => $this->getDataProviderPrices()
  129. ]);
  130. }
  131. public function getDataProviderPrices()
  132. {
  133. return new ActiveDataProvider([
  134. 'query' => $this->getProducerPriceRangeManager()->queryProducerPriceRanges()->query(),
  135. 'pagination' => [
  136. 'pageSize' => 100,
  137. ],
  138. ]);
  139. }
  140. /**
  141. * Liste les producteurs utilisant la plateforme.
  142. */
  143. public function actionProducers()
  144. {
  145. $dataProviderProducers = new ActiveDataProvider([
  146. 'query' => $this->getProducerManager()->queryProducersActive()->query(),
  147. 'pagination' => [
  148. 'pageSize' => 100,
  149. ],
  150. ]);
  151. return $this->render('producers', [
  152. 'dataProviderProducers' => $dataProviderProducers,
  153. 'producersArray' => $this->getProducerManager()->findProducersActive()
  154. ]);
  155. }
  156. public function actionAbout()
  157. {
  158. $aboutFewNumbers = Yii::$app->cache->getOrSet('about_few_numbers4', function () {
  159. $producerManager = $this->getProducerManager();
  160. $pointSaleManager = $this->getPointSaleManager();
  161. $userManager = $this->getUserManager();
  162. $orderManager = $this->getOrderManager();
  163. $countProducersActive = $producerManager->countProducersActiveWithTurnover();
  164. $countPointSalesActive = $pointSaleManager->countPointSalesActiveLastThreeMonths();
  165. $countUsersActive = $userManager->countUsersActiveLastThreeMonths();
  166. $averageOrdersPerDay = $orderManager->countGlobalUserOrdersAverageLastSevenDays();
  167. $averageTurnover = $orderManager->getAverageTurnoverLastThreeMonths();
  168. $resultMatomoApiVisitSummary = json_decode(file_get_contents(Yii::$app->params['matomoApiVisitSummaryUrl']));
  169. $numberVisitsMonth = $resultMatomoApiVisitSummary->nb_uniq_visitors;
  170. return $this->renderPartial('_about_few_numbers', [
  171. 'countProducersActive' => $countProducersActive,
  172. 'countPointSalesActive' => $countPointSalesActive,
  173. 'countUsersActive' => $countUsersActive,
  174. 'averageOrdersPerDay' => $averageOrdersPerDay,
  175. 'averageTurnover' => $averageTurnover,
  176. 'numberVisitsMonth' => $numberVisitsMonth
  177. ]);
  178. }, 60 * 60 * 24);
  179. $producerManager = $this->getProducerManager();
  180. return $this->render('about', [
  181. 'countProducers' => $producerManager->countProducersActiveWithTurnover(),
  182. 'aboutFewNumbers' => $aboutFewNumbers
  183. ]);
  184. }
  185. public function actionSourceCode()
  186. {
  187. return $this->render('source_code');
  188. }
  189. /**
  190. * Liste les producteurs utilisant la plateforme.
  191. */
  192. public function actionAjaxProducers()
  193. {
  194. \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
  195. $producersArray = $this->getProducerManager()->findProducersActive();
  196. $producersArrayReturn = [];
  197. foreach($producersArray as $producer) {
  198. $producersArrayReturn[] = [
  199. 'id' => $producer->id,
  200. 'name' => Html::encode($producer->name),
  201. 'type' => Html::encode($producer->type),
  202. 'address' => Html::encode($producer->postcode.' '.$producer->city),
  203. 'latitude' => $producer->latitude,
  204. 'longitude' => $producer->longitude,
  205. 'link' => $this->getUrlManagerProducer()->createAbsoluteUrl(['site/index', 'slug_producer' => $producer->slug])
  206. ];
  207. }
  208. return $producersArrayReturn;
  209. }
  210. /**
  211. * Affiche la page de connexion et traite le formulaire de connexion.
  212. */
  213. public function actionLogin()
  214. {
  215. if (!\Yii::$app->user->isGuest) {
  216. return \Yii::$app->getResponse()->redirect(['site/index']);
  217. }
  218. $model = new LoginForm();
  219. if ($model->load(Yii::$app->request->post()) && $model->login()) {
  220. $returnUrl = \Yii::$app->request->get('return_url');
  221. if ($returnUrl) {
  222. return $this->redirect($returnUrl);
  223. }
  224. else {
  225. $userProducerArray = $this->getUserProducerManager()->findUserProducersByUser(GlobalParam::getCurrentUser());
  226. if ($userProducerArray
  227. && is_array($userProducerArray)
  228. && count($userProducerArray) == 1) {
  229. $urlRedirect = $this->getUrlManagerProducer()
  230. ->createAbsoluteUrl([
  231. 'site/index',
  232. 'slug_producer' => $userProducerArray[0]->producer->slug
  233. ]);
  234. return $this->redirect($urlRedirect);
  235. }
  236. else {
  237. return $this->goBack();
  238. }
  239. }
  240. } else {
  241. return $this->render('@frontend/views/site/login', [
  242. 'model' => $model,
  243. ]);
  244. }
  245. }
  246. /**
  247. * Déconnecte l'utilisateur.
  248. */
  249. public function actionLogout()
  250. {
  251. \Yii::$app->user->logout();
  252. return $this->goHome();
  253. }
  254. /**
  255. * Affiche la page de contact et traite le formulaire s'il est soumis.
  256. */
  257. public function actionContact()
  258. {
  259. $model = new ContactForm();
  260. if ($model->load(Yii::$app->request->post()) && $model->validate()) {
  261. if ($model->sendEmail(Yii::$app->params['adminEmail'])) {
  262. $this->setFlash('success', 'Votre message a bien été envoyé. Nous vous répondrons dès que possible.');
  263. }
  264. else {
  265. $this->setFlash('error', 'Il y a eu une erreur lors de l\'envoi de votre message.');
  266. }
  267. return $this->refresh();
  268. }
  269. else {
  270. return $this->render('contact', [
  271. 'model' => $model,
  272. ]);
  273. }
  274. }
  275. /**
  276. * Affiche la page d'inscription et traite son formulaire.
  277. */
  278. public function actionSignup()
  279. {
  280. $model = new SignupForm();
  281. $producerManager = $this->getProducerManager();
  282. if ($model->load(Yii::$app->request->post())) {
  283. $user = $model->signup();
  284. if ($user && Yii::$app->getUser()->login($user)) {
  285. if ($model->isProducer()) {
  286. $this->redirect(['site/signup-confirm']);
  287. }
  288. else {
  289. $producer = $producerManager->findOneProducerById($model->id_producer);
  290. if ($producer) {
  291. $this->redirect(['site/signup-confirm', 'idProducerRedirect' => $producer->id]);
  292. }
  293. else {
  294. $this->redirect(['site/index']);
  295. }
  296. }
  297. }
  298. }
  299. // Liste des producteurs disponibles
  300. $producersArray = $producerManager->populateProducerDropdown();
  301. $dataProducers = $producersArray['data'];
  302. $optionsProducers = $producersArray['options'];
  303. return $this->render('signup', [
  304. 'model' => $model,
  305. 'dataProducers' => $dataProducers,
  306. 'dataProviderPrices' => $this->getDataProviderPrices(),
  307. 'optionsProducers' => $optionsProducers,
  308. ]);
  309. }
  310. public function actionSignupConfirm($idProducerRedirect = null)
  311. {
  312. $producerManager = $this->getProducerManager();
  313. $user = $this->getUserCurrent();
  314. $producerRedirect = $idProducerRedirect ? $producerManager->findOneProducerById($idProducerRedirect) : null ;
  315. if(!$user) {
  316. throw new NotFoundHttpException('Page introuvable');
  317. }
  318. return $this->render('signup_confirm', [
  319. 'user' => $user,
  320. 'producerRedirect' => $producerRedirect
  321. ]);
  322. }
  323. /**
  324. * Affiche la page de demande de nouveau mot de passe.
  325. * Traitement du formulaire.
  326. */
  327. public function actionRequestPasswordReset()
  328. {
  329. $model = new PasswordResetRequestForm();
  330. if ($model->load(Yii::$app->request->post()) && $model->validate()) {
  331. if ($model->sendEmail()) {
  332. $this->setFlash('success', 'Un lien vous permettant de réinitialiser votre mot de passe
  333. vient d\'être envoyé sur votre boîte mail.');
  334. return $this->goHome();
  335. }
  336. else {
  337. $this->setFlash('error', 'Sorry, we are unable to reset password for email provided.');
  338. }
  339. }
  340. return $this->render('requestPasswordResetToken', [
  341. 'model' => $model,
  342. ]);
  343. }
  344. /**
  345. * Met à jour le mot de passe de l'utilisateur.
  346. */
  347. public function actionResetPassword($token)
  348. {
  349. try {
  350. $model = new ResetPasswordForm($token);
  351. } catch (InvalidParamException $e) {
  352. throw new BadRequestHttpException($e->getMessage());
  353. }
  354. if ($model->load($this->getRequest()->post())
  355. && $model->validate()
  356. && $model->resetPassword()) {
  357. $this->setFlash('success', 'Votre nouveau mot de passe vient d\'être sauvegardé.');
  358. return $this->goHome();
  359. }
  360. return $this->render('resetPassword', [
  361. 'model' => $model,
  362. ]);
  363. }
  364. /**
  365. * Affiche le formulaire de demande de code pour accéder à certains producteurs.
  366. */
  367. public function actionProducerCode(int $id)
  368. {
  369. $producerManager = $this->getProducerManager();
  370. $producer = $producerManager->findOneProducerById($id);
  371. if (!$producer) {
  372. throw new \yii\web\HttpException(404, 'Producteur introuvable');
  373. }
  374. $producerCodeForm = new ProducerCodeForm();
  375. $producerCodeForm->id_producer = $id;
  376. if ($producerCodeForm->load($this->getRequest()->post())
  377. && $producerCodeForm->validate()) {
  378. $this->getLogic()->setProducerContext($producer);
  379. $producerManager->addUser(GlobalParam::getCurrentUser(), $producer);
  380. $this->redirect($this->getUrlManagerProducer()->createAbsoluteUrl(['site/index', 'slug_producer' => $producer->slug]));
  381. }
  382. return $this->render('producer_code', [
  383. 'producer' => $producer,
  384. 'producerCodeForm' => $producerCodeForm,
  385. ]);
  386. }
  387. /**
  388. * Affiche la page de connexion / inscription pour accéder notamment au
  389. * formulaire de commande des producteurs.
  390. */
  391. public function actionProducer(int $id)
  392. {
  393. $loginForm = new LoginForm();
  394. $signupForm = new SignupForm();
  395. $producerManager = $this->getProducerManager();
  396. $producer = $producerManager->findOneProducerById($id);
  397. $this->getLogic()->setProducerContext($producer);
  398. $loginForm->id_producer = $id;
  399. $signupForm->id_producer = $id;
  400. $signupForm->option_user_producer = 'user';
  401. $returnUrl = $this->getRequest()->get('returnUrl', $this->getUrlManagerProducer()->createAbsoluteUrl(['site/index', 'slug_producer' => $producer->slug]));
  402. if (Yii::$app->user->isGuest) {
  403. if ($loginForm->load($this->getRequest()->post()) && $loginForm->login()) {
  404. if (!strlen($producer->code)) {
  405. $producerManager->addUser(GlobalParam::getCurrentUser(), $producer);
  406. }
  407. $this->redirect($returnUrl);
  408. }
  409. if ($signupForm->load($this->getRequest()->post())
  410. && ($user = $signupForm->signup())
  411. && Yii::$app->user->login($user)) {
  412. $this->redirect($returnUrl);
  413. }
  414. }
  415. else {
  416. $this->redirect($returnUrl);
  417. }
  418. return $this->render('producer', [
  419. 'loginForm' => $loginForm,
  420. 'signupForm' => $signupForm,
  421. 'producer' => $producer,
  422. ]);
  423. }
  424. /**
  425. * Indique à l'utilisateur que l'espace d'un producteur est hors ligne.
  426. */
  427. public function actionProducerOffline(int $id)
  428. {
  429. return $this->render('producer_offline', [
  430. 'producer' => $this->getProducerManager()->findOneProducerById($id),
  431. ]);
  432. }
  433. /**
  434. * Affiche les mentions légales.
  435. */
  436. public function actionMentions()
  437. {
  438. return $this->render('mentions');
  439. }
  440. /**
  441. * Affiche les conditions générale de service.
  442. */
  443. public function actionCgv()
  444. {
  445. return $this->render('cgv');
  446. }
  447. /**
  448. * Affiche les précisions concernant l'utilisation du crédit.
  449. */
  450. public function actionCredit()
  451. {
  452. return $this->render('credit');
  453. }
  454. }