X-Git-Url: https://git.rapsys.eu/airbundle/blobdiff_plain/a28f854a99affabca1b5ced856dec3469b62369f..361cd87861355561d4d5d4005753b48f1fc5f526:/Controller/DefaultController.php?ds=sidebyside diff --git a/Controller/DefaultController.php b/Controller/DefaultController.php index def8c54..49ad3c3 100644 --- a/Controller/DefaultController.php +++ b/Controller/DefaultController.php @@ -2,71 +2,142 @@ namespace Rapsys\AirBundle\Controller; -use Rapsys\AirBundle\Entity\Application; -use Rapsys\AirBundle\Entity\Session; use Symfony\Bridge\Twig\Mime\TemplatedEmail; -use Symfony\Bundle\FrameworkBundle\Controller\AbstractController; +use Symfony\Component\DependencyInjection\ContainerAwareTrait; use Symfony\Component\DependencyInjection\ContainerInterface; use Symfony\Component\Form\FormError; use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\HttpFoundation\RequestStack; +use Symfony\Component\HttpFoundation\Response; use Symfony\Component\Mailer\Exception\TransportExceptionInterface; use Symfony\Component\Mailer\MailerInterface; use Symfony\Component\Mime\Address; -use Symfony\Component\Routing\Generator\UrlGeneratorInterface; use Symfony\Component\Routing\RouterInterface; use Symfony\Component\Translation\TranslatorInterface; -use Rapsys\UserBundle\Utils\Slugger; + +use Rapsys\AirBundle\Entity\Location; +use Rapsys\AirBundle\Entity\Session; +use Rapsys\AirBundle\Pdf\DisputePdf; class DefaultController extends AbstractController { - //Config array + ///Config array protected $config; - //Context array + ///ContainerInterface instance + protected $container; + + ///Context array protected $context; - //Router instance + ///Router instance protected $router; - //Slugger instance - protected $slugger; - - //Translator instance + ///Translator instance protected $translator; + ///Request instance + protected $request; + + ///Locale instance + protected $locale; + /** - * Inject container and translator interface + * Inject container, router and translator interface * * @param ContainerInterface $container The container instance * @param RouterInterface $router The router instance - * @param Slugger $slugger The slugger instance * @param TranslatorInterface $translator The translator instance */ - public function __construct(ContainerInterface $container, RouterInterface $router, Slugger $slugger, TranslatorInterface $translator) { + public function __construct(ContainerInterface $container, RouterInterface $router, TranslatorInterface $translator) { //Retrieve config - $this->config = $container->getParameter($this->getAlias()); + $this->config = $container->getParameter(self::getAlias()); + + //Set the container + $this->container = $container; //Set the router $this->router = $router; - //Set the slugger - $this->slugger = $slugger; - //Set the translator $this->translator = $translator; //Set the context $this->context = [ - 'copy_long' => $translator->trans($this->config['copy']['long']), - 'copy_short' => $translator->trans($this->config['copy']['short']), - 'site_ico' => $this->config['site']['ico'], - 'site_logo' => $this->config['site']['logo'], - 'site_png' => $this->config['site']['png'], - 'site_svg' => $this->config['site']['svg'], - 'site_title' => $translator->trans($this->config['site']['title']), - 'site_url' => $router->generate($this->config['site']['url']) + 'contact' => [ + 'title' => $translator->trans($this->config['contact']['title']), + 'mail' => $this->config['contact']['mail'] + ], + 'copy' => [ + 'by' => $translator->trans($this->config['copy']['by']), + 'link' => $this->config['copy']['link'], + 'long' => $translator->trans($this->config['copy']['long']), + 'short' => $translator->trans($this->config['copy']['short']), + 'title' => $this->config['copy']['title'] + ], + 'page' => [ + 'description' => null, + 'section' => null, + 'title' => null + ], + 'site' => [ + 'donate' => $this->config['site']['donate'], + 'ico' => $this->config['site']['ico'], + 'logo' => $this->config['site']['logo'], + 'png' => $this->config['site']['png'], + 'svg' => $this->config['site']['svg'], + 'title' => $translator->trans($this->config['site']['title']), + 'url' => $router->generate($this->config['site']['url']) + ], + 'canonical' => null, + 'alternates' => [], + 'facebook' => [ + 'heads' => [ + 'og' => 'http://ogp.me/ns#', + 'fb' => 'http://ogp.me/ns/fb#' + ], + 'metas' => [ + 'og:type' => 'article', + 'og:site_name' => $this->translator->trans($this->config['site']['title']), + #'fb:admins' => $this->config['facebook']['admins'], + 'fb:app_id' => $this->config['facebook']['apps'] + ], + 'texts' => [] + ], + 'forms' => [] ]; } + /** + * The about page + * + * @desc Display the about informations + * + * @param Request $request The request instance + * @return Response The rendered view + */ + public function about(Request $request): Response { + //Set page + $this->context['page']['title'] = $this->translator->trans('About'); + + //Set description + $this->context['page']['description'] = $this->translator->trans('Libre Air about'); + + //Set keywords + $this->context['keywords'] = [ + $this->translator->trans('about'), + $this->translator->trans('Libre Air') + ]; + + //Render template + $response = $this->render('@RapsysAir/default/about.html.twig', $this->context); + $response->setEtag(md5($response->getContent())); + $response->setPublic(); + $response->isNotModified($request); + + //Return response + return $response; + } + /** * The contact page * @@ -77,12 +148,21 @@ class DefaultController extends AbstractController { * * @return Response The rendered view or redirection */ - public function contact(Request $request, MailerInterface $mailer) { - //Set section - $section = $this->translator->trans('Contact'); + public function contact(Request $request, MailerInterface $mailer): Response { + //Set page + $this->context['page']['title'] = $this->translator->trans('Contact'); + + //Set description + $this->context['page']['description'] = $this->translator->trans('Contact Libre Air'); - //Set title - $title = $section.' - '.$this->translator->trans($this->config['site']['title']); + //Set keywords + $this->context['keywords'] = [ + $this->translator->trans('contact'), + $this->translator->trans('Libre Air'), + $this->translator->trans('outdoor'), + $this->translator->trans('Argentine Tango'), + $this->translator->trans('calendar') + ]; //Create the form according to the FormType created previously. //And give the proper parameters @@ -104,8 +184,7 @@ class DefaultController extends AbstractController { //Set sender ->from(new Address($data['mail'], $data['name'])) //Set recipient - //XXX: remove the debug set in vendor/symfony/mime/Address.php +46 - ->to(new Address($this->config['contact']['mail'], $this->config['contact']['name'])) + ->to(new Address($this->context['contact']['mail'], $this->context['contact']['title'])) //Set subject ->subject($data['subject']) @@ -133,53 +212,303 @@ class DefaultController extends AbstractController { } catch(TransportExceptionInterface $e) { if ($message = $e->getMessage()) { //Add error message mail unreachable - $form->get('mail')->addError(new FormError($this->translator->trans('Unable to contact: %mail%: %message%', ['%mail%' => $this->config['contact']['mail'], '%message%' => $this->translator->trans($message)]))); + $form->get('mail')->addError(new FormError($this->translator->trans('Unable to contact: %mail%: %message%', ['%mail%' => $this->context['contact']['mail'], '%message%' => $this->translator->trans($message)]))); } else { //Add error message mail unreachable - $form->get('mail')->addError(new FormError($this->translator->trans('Unable to contact: %mail%', ['%mail%' => $this->config['contact']['mail']]))); + $form->get('mail')->addError(new FormError($this->translator->trans('Unable to contact: %mail%', ['%mail%' => $this->context['contact']['mail']]))); } } } } //Render template - return $this->render('@RapsysAir/form/contact.html.twig', ['title' => $title, 'section' => $section, 'form' => $form->createView(), 'sent' => $request->query->get('sent', 0)]+$this->context); + return $this->render('@RapsysAir/form/contact.html.twig', ['form' => $form->createView(), 'sent' => $request->query->get('sent', 0)]+$this->context); + } + + /** + * The dispute page + * + * @desc Generate a dispute document + * + * @param Request $request The request instance + * @param MailerInterface $mailer The mailer instance + * + * @return Response The rendered view or redirection + */ + public function dispute(Request $request, MailerInterface $mailer): Response { + //Prevent non-guest to access here + $this->denyAccessUnlessGranted('ROLE_USER', null, $this->translator->trans('Unable to access this page without role %role%!', ['%role%' => $this->translator->trans('User')])); + + //Set page + $this->context['page']['title'] = $this->translator->trans('Dispute'); + + //Set description + $this->context['page']['description'] = $this->translator->trans('Libre Air dispute'); + + //Set keywords + $this->context['keywords'] = [ + $this->translator->trans('dispute'), + $this->translator->trans('Libre Air'), + $this->translator->trans('outdoor'), + $this->translator->trans('Argentine Tango'), + $this->translator->trans('calendar') + ]; + + //Create the form according to the FormType created previously. + //And give the proper parameters + $form = $this->createForm('Rapsys\AirBundle\Form\DisputeType', ['court' => 'Paris', 'abstract' => 'Pour constater cette prétendue infraction, les agents verbalisateurs ont pénétré dans un jardin privatif, sans visibilité depuis la voie publique, situé derrière un batiment privé, pour ce faire ils ont franchi au moins un grillage de chantier ou des potteaux métalliques séparant le terrain privé de la voie publique de l\'autre côté du batiment.'], [ + 'action' => $this->generateUrl('rapsys_air_dispute'), + 'method' => 'POST' + ]); + + if ($request->isMethod('POST')) { + // Refill the fields in case the form is not valid. + $form->handleRequest($request); + + if ($form->isValid()) { + //Get data + $data = $form->getData(); + + //Gathering offense + if (!empty($data['offense']) && $data['offense'] == 'gathering') { + //Add gathering + $output = DisputePdf::genGathering($data['court'], $data['notice'], $data['agent'], $data['service'], $data['abstract'], $this->translator->trans($this->getUser()->getCivility()->getTitle()), $this->getUser()->getForename(), $this->getUser()->getSurname()); + //Traffic offense + } elseif (!empty($data['offense'] && $data['offense'] == 'traffic')) { + //Add traffic + $output = DisputePdf::genTraffic($data['court'], $data['notice'], $data['agent'], $data['service'], $data['abstract'], $this->translator->trans($this->getUser()->getCivility()->getTitle()), $this->getUser()->getForename(), $this->getUser()->getSurname()); + //Unsupported offense + } else { + header('Content-Type: text/plain'); + die('TODO'); + exit; + } + + //Send common headers + header('Content-Type: application/pdf'); + + //Send remaining headers + header('Cache-Control: private, max-age=0, must-revalidate'); + header('Pragma: public'); + + //Send content-length + header('Content-Length: '.strlen($output)); + + //Display the pdf + echo $output; + + //Die for now + exit; + +# //Create message +# $message = (new TemplatedEmail()) +# //Set sender +# ->from(new Address($data['mail'], $data['name'])) +# //Set recipient +# //XXX: remove the debug set in vendor/symfony/mime/Address.php +46 +# ->to(new Address($this->config['contact']['mail'], $this->config['contact']['title'])) +# //Set subject +# ->subject($data['subject']) +# +# //Set path to twig templates +# ->htmlTemplate('@RapsysAir/mail/contact.html.twig') +# ->textTemplate('@RapsysAir/mail/contact.text.twig') +# +# //Set context +# ->context( +# [ +# 'subject' => $data['subject'], +# 'message' => strip_tags($data['message']), +# ]+$this->context +# ); +# +# //Try sending message +# //XXX: mail delivery may silently fail +# try { +# //Send message +# $mailer->send($message); +# +# //Redirect on the same route with sent=1 to cleanup form +# return $this->redirectToRoute($request->get('_route'), ['sent' => 1]+$request->get('_route_params')); +# //Catch obvious transport exception +# } catch(TransportExceptionInterface $e) { +# if ($message = $e->getMessage()) { +# //Add error message mail unreachable +# $form->get('mail')->addError(new FormError($this->translator->trans('Unable to contact: %mail%: %message%', ['%mail%' => $this->config['contact']['mail'], '%message%' => $this->translator->trans($message)]))); +# } else { +# //Add error message mail unreachable +# $form->get('mail')->addError(new FormError($this->translator->trans('Unable to contact: %mail%', ['%mail%' => $this->config['contact']['mail']]))); +# } +# } + } + } + + //Render template + return $this->render('@RapsysAir/default/dispute.html.twig', ['form' => $form->createView(), 'sent' => $request->query->get('sent', 0)]+$this->context); } /** * The index page * - * @desc Welcome the user + * @desc Display all granted sessions with an application or login form + * + * @param Request $request The request instance + * @return Response The rendered view + */ + public function index(Request $request): Response { + //Fetch doctrine + $doctrine = $this->getDoctrine(); + + //Set page + $this->context['page']['title'] = $this->translator->trans('Argentine Tango in Paris'); + + //Set description + $this->context['page']['description'] = $this->translator->trans('Outdoor Argentine Tango session calendar in Paris'); + + //Set keywords + $this->context['keywords'] = [ + $this->translator->trans('Argentine Tango'), + $this->translator->trans('Paris'), + $this->translator->trans('outdoor'), + $this->translator->trans('calendar'), + $this->translator->trans('Libre Air') + ]; + + //Set facebook type + //XXX: only valid for home page + $this->context['facebook']['metas']['og:type'] = 'website'; + + //Compute period + $period = new \DatePeriod( + //Start from first monday of week + new \DateTime('Monday this week'), + //Iterate on each day + new \DateInterval('P1D'), + //End with next sunday and 4 weeks + new \DateTime( + $this->isGranted('IS_AUTHENTICATED_REMEMBERED')?'Monday this week + 3 week':'Monday this week + 2 week' + ) + ); + + //Fetch calendar + $calendar = $doctrine->getRepository(Session::class)->fetchCalendarByDatePeriod($this->translator, $period, null, $request->get('session'), !$this->isGranted('IS_AUTHENTICATED_REMEMBERED')); + + //Fetch locations + //XXX: we want to display all active locations anyway + $locations = $doctrine->getRepository(Location::class)->findTranslatedSortedByPeriod($this->translator, $period); + + //Render the view + return $this->render('@RapsysAir/default/index.html.twig', ['calendar' => $calendar, 'locations' => $locations]+$this->context); + + //Set Cache-Control must-revalidate directive + //TODO: add a javascript forced refresh after 1h ? or header refresh ? + #$response->setPublic(true); + #$response->setMaxAge(300); + #$response->mustRevalidate(); + ##$response->setCache(['public' => true, 'max_age' => 300]); + + //Return the response + #return $response; + } + + /** + * The organizer regulation page + * + * @desc Display the organizer regulation policy + * + * @param Request $request The request instance + * @return Response The rendered view + */ + public function organizerRegulation(Request $request): Response { + //Set page + $this->context['page']['title'] = $this->translator->trans('Organizer regulation'); + + //Set description + $this->context['page']['description'] = $this->translator->trans('Libre Air organizer regulation'); + + //Set keywords + $this->context['keywords'] = [ + $this->translator->trans('organizer regulation'), + $this->translator->trans('Libre Air') + ]; + + //Render template + $response = $this->render('@RapsysAir/default/organizer_regulation.html.twig', $this->context); + + //Set as cachable + $response->setEtag(md5($response->getContent())); + $response->setPublic(); + $response->isNotModified($request); + + //Return response + return $response; + } + + /** + * The terms of service page + * + * @desc Display the terms of service policy * + * @param Request $request The request instance * @return Response The rendered view */ - public function index() { - //Set section - $section = $this->translator->trans('Index'); + public function termsOfService(Request $request): Response { + //Set page + $this->context['page']['title'] = $this->translator->trans('Terms of service'); - //Set title - $title = $section.' - '.$this->context['site_title']; + //Set description + $this->context['page']['description'] = $this->translator->trans('Libre Air terms of service'); + + //Set keywords + $this->context['keywords'] = [ + $this->translator->trans('terms of service'), + $this->translator->trans('Libre Air') + ]; //Render template - return $this->render('@RapsysAir/default/index.html.twig', ['title' => $title, 'section' => $section]+$this->context); + $response = $this->render('@RapsysAir/default/terms_of_service.html.twig', $this->context); + + //Set as cachable + $response->setEtag(md5($response->getContent())); + $response->setPublic(); + $response->isNotModified($request); + + //Return response + return $response; } /** - * The regulation page + * The frequently asked questions page * - * @desc Display the regulation policy + * @desc Display the frequently asked questions * + * @param Request $request The request instance * @return Response The rendered view */ - public function regulation() { - //Set section - $section = $this->translator->trans('Regulation'); + public function frequentlyAskedQuestions(Request $request): Response { + //Set page + $this->context['page']['title'] = $this->translator->trans('Frequently asked questions'); + + //Set description + $this->context['page']['description'] = $this->translator->trans('Libre Air frequently asked questions'); - //Set title - $title = $section.' - '.$this->context['site_title']; + //Set keywords + $this->context['keywords'] = [ + $this->translator->trans('frequently asked questions'), + $this->translator->trans('faq'), + $this->translator->trans('Libre Air') + ]; //Render template - return $this->render('@RapsysAir/default/regulation.html.twig', ['title' => $title, 'section' => $section]+$this->context); + $response = $this->render('@RapsysAir/default/frequently_asked_questions.html.twig', $this->context); + + //Set as cachable + $response->setEtag(md5($response->getContent())); + $response->setPublic(); + $response->isNotModified($request); + + //Return response + return $response; } /** @@ -187,7 +516,17 @@ class DefaultController extends AbstractController { * * {@inheritdoc} */ - public function getAlias() { + public function getAlias(): string { return 'rapsys_air'; } + + /** + * Renders a view + * + * {@inheritdoc} + */ + /*protected function render(string $view, array $parameters = [], Response $response = null): Response { + //Call parent render + return parent::render($view, $parameters, $response); + }*/ }