-<?php
+<?php declare(strict_types=1);
+
+/*
+ * This file is part of the Rapsys AirBundle package.
+ *
+ * (c) Raphaël Gertz <symfony@rapsys.eu>
+ *
+ * For the full copyright and license information, please view the LICENSE
+ * file that was distributed with this source code.
+ */
namespace Rapsys\AirBundle\Controller;
+use Doctrine\ORM\NoResultException;
+use Doctrine\ORM\ORMInvalidArgumentException;
+use Symfony\Component\Form\FormError;
use Symfony\Component\HttpFoundation\Request;
+use Symfony\Component\HttpFoundation\Response;
+use Symfony\Component\Routing\Exception\MethodNotAllowedException;
+use Symfony\Component\Routing\Exception\ResourceNotFoundException;
+use Symfony\Component\Routing\RequestContext;
+
+use Rapsys\AirBundle\Entity\Application;
+use Rapsys\AirBundle\Entity\Dance;
+use Rapsys\AirBundle\Entity\Location;
+use Rapsys\AirBundle\Entity\Session;
use Rapsys\AirBundle\Entity\Slot;
use Rapsys\AirBundle\Entity\User;
-use Rapsys\AirBundle\Entity\Session;
-use Rapsys\AirBundle\Entity\Application;
-class ApplicationController extends DefaultController {
+/**
+ * {@inheritdoc}
+ */
+class ApplicationController extends AbstractController {
/**
* Add application
*
* @desc Persist application and all required dependencies in database
*
* @param Request $request The request instance
+ * @param Registry $manager The doctrine registry
+ * @param EntityManagerInterface $manager The doctrine entity manager
*
* @return Response The rendered view or redirection
*
* @throws \RuntimeException When user has not at least guest role
*/
public function add(Request $request) {
- //Prevent non-guest to access here
- $this->denyAccessUnlessGranted('ROLE_GUEST', null, $this->translator->trans('Unable to access this page without role %role%!', ['%role%' => $this->translator->trans('Guest')]));
+ //Without guest role
+ if (!$this->checker->isGranted('ROLE_GUEST')) {
+ //Throw 403
+ throw $this->createAccessDeniedException($this->translator->trans('Unable to access this page without role %role%!', ['%role%' => $this->translator->trans('Guest')]));
+ }
+
+ //Get favorites dances
+ $danceFavorites = $this->doctrine->getRepository(Dance::class)->findByUserId($this->security->getUser()->getId());
+
+ //Set dance default
+ $danceDefault = !empty($danceFavorites)?current($danceFavorites):null;
+
+
+ //Get favorites locations
+ $locationFavorites = $this->doctrine->getRepository(Location::class)->findByUserId($this->security->getUser()->getId());
+
+ //Set location default
+ $locationDefault = !empty($locationFavorites)?current($locationFavorites):null;
+
+ //With admin
+ if ($this->checker->isGranted('ROLE_ADMIN')) {
+ //Get dances
+ $dances = $this->doctrine->getRepository(Dance::class)->findAll();
+
+ //Get locations
+ $locations = $this->doctrine->getRepository(Location::class)->findAll();
+ //Without admin
+ } else {
+ //Restrict to favorite dances
+ $dances = $danceFavorites;
+
+ //Reset favorites
+ $danceFavorites = [];
+
+ //Restrict to favorite locations
+ $locations = $locationFavorites;
+
+ //Reset favorites
+ $locationFavorites = [];
+ }
//Create ApplicationType form
- $form = $this->createForm('Rapsys\AirBundle\Form\ApplicationType', null, [
+ $form = $this->factory->create('Rapsys\AirBundle\Form\ApplicationType', null, [
//Set the action
'action' => $this->generateUrl('rapsys_air_application_add'),
//Set the form attribute
#'attr' => [ 'class' => 'col' ],
- //Set admin
- 'admin' => $this->isGranted('ROLE_ADMIN'),
+ //Set dance choices
+ 'dance_choices' => $dances,
+ //Set dance default
+ 'dance_default' => $danceDefault,
+ //Set dance favorites
+ 'dance_favorites' => $danceFavorites,
+ //Set location choices
+ 'location_choices' => $locations,
+ //Set location default
+ 'location_default' => $locationDefault,
+ //Set location favorites
+ 'location_favorites' => $locationFavorites,
+ //With user
+ 'user' => $this->checker->isGranted('ROLE_ADMIN'),
+ //Set user choices
+ 'user_choices' => $this->doctrine->getRepository(User::class)->findChoicesAsArray(),
//Set default user to current
- 'user' => $this->getUser()->getId(),
+ 'user_default' => $this->security->getUser()->getId(),
//Set default slot to evening
//XXX: default to Evening (3)
- 'slot' => $this->getDoctrine()->getRepository(Slot::class)->findOneById(3)
+ 'slot_default' => $this->doctrine->getRepository(Slot::class)->findOneByTitle('Evening')
]);
- //Reject non post requests
- if (!$request->isMethod('POST')) {
- throw new \RuntimeException('Request method MUST be POST');
- }
-
//Refill the fields in case of invalid form
$form->handleRequest($request);
//Handle invalid form
- if (!$form->isValid()) {
- //Set section
- $section = $this->translator->trans('Application Add');
-
+ if (!$form->isSubmitted() || !$form->isValid()) {
//Set title
- $title = $section.' - '.$this->translator->trans($this->config['site']['title']);
+ $title = $this->translator->trans('Application add');
//Render the view
- return $this->render('@RapsysAir/application/add.html.twig', ['title' => $title, 'section' => $section, 'form' => $form]+$this->context);
+ return $this->render('@RapsysAir/application/add.html.twig', ['title' => $title, 'form' => $form->createView()]+$this->context);
}
- //Get doctrine
- $doctrine = $this->getDoctrine();
-
- //Get manager
- $manager = $doctrine->getManager();
-
//Get data
$data = $form->getData();
//Protect session fetching
try {
//Fetch session
- $session = $doctrine->getRepository(Session::class)->findOneByLocationSlotDate($data['location'], $data['slot'], $data['date']);
+ $session = $this->doctrine->getRepository(Session::class)->findOneByLocationSlotDate($data['location'], $data['slot'], $data['date']);
//Catch no session case
- } catch (\Doctrine\ORM\NoResultException $e) {
+ } catch (NoResultException $e) {
//Create the session
$session = new Session();
$session->setLocation($data['location']);
$session->setDate($data['date']);
$session->setSlot($data['slot']);
- $session->setCreated(new \DateTime('now'));
- $session->setUpdated(new \DateTime('now'));
+
+ //Get location
+ $location = $data['location']->getTitle();
+
+ //Get slot
+ $slot = $data['slot']->getTitle();
+
+ //Get premium
+ //XXX: premium is stored only for Afternoon and Evening
+ $premium = $session->isPremium();
+
+ //Set default length at 6h
+ //XXX: date part will be truncated on save
+ $session->setLength(new \DateTime('06:00:00'));
+
+ //Check if admin
+ if ($this->checker->isGranted('ROLE_ADMIN')) {
+ //Check if morning
+ if ($slot == 'Morning') {
+ //Set begin at 9h
+ $session->setBegin(new \DateTime('09:00:00'));
+
+ //Set length at 5h
+ $session->setLength(new \DateTime('05:00:00'));
+ //Check if afternoon
+ } elseif ($slot == 'Afternoon') {
+ //Set begin at 18h
+ $session->setBegin(new \DateTime('15:30:00'));
+
+ //Set length at 5h
+ $session->setLength(new \DateTime('05:30:00'));
+ //Check if evening
+ } elseif ($slot == 'Evening') {
+ //Set begin at 19h00
+ $session->setBegin(new \DateTime('19:30:00'));
+
+ //Set length at 5h
+ $session->setLength(new \DateTime('05:30:00'));
+
+ //Check if next day is premium
+ if ($premium) {
+ //Set length at 7h
+ $session->setLength(new \DateTime('06:30:00'));
+ }
+ //Check if after
+ } else {
+ //Set begin at 1h
+ $session->setBegin(new \DateTime('01:00:00'));
+
+ //Set length at 4h
+ $session->setLength(new \DateTime('04:00:00'));
+
+ //Check if next day is premium
+ if ($premium) {
+ //Set begin at 2h
+ $session->setBegin(new \DateTime('02:00:00'));
+
+ //Set length at 3h
+ $session->setLength(new \DateTime('03:00:00'));
+ }
+ }
+ //Tino-Rossi garden => 14h -> 19h | 19h -> 01/02h
+ } elseif (in_array($location, ['Tino-Rossi garden']) && in_array($slot, ['Afternoon', 'Evening', 'After'])) {
+ //Check if afternoon
+ if ($slot == 'Afternoon') {
+ //Set begin at 14h
+ $session->setBegin(new \DateTime('14:00:00'));
+
+ //Set length at 5h
+ $session->setLength(new \DateTime('05:00:00'));
+ //Check if evening
+ } elseif ($slot == 'Evening') {
+ //Set begin at 19h
+ $session->setBegin(new \DateTime('19:00:00'));
+
+ //Check if next day is premium
+ if ($premium) {
+ //Set length at 7h
+ $session->setLength(new \DateTime('07:00:00'));
+ }
+ //Check if after
+ } else {
+ //Set begin at 1h
+ $session->setBegin(new \DateTime('01:00:00'));
+
+ //Set length at 4h
+ $session->setLength(new \DateTime('04:00:00'));
+
+ //Check if next day is premium
+ if ($premium) {
+ //Set begin at 2h
+ $session->setBegin(new \DateTime('02:00:00'));
+
+ //Set length at 3h
+ $session->setLength(new \DateTime('03:00:00'));
+ }
+ }
+ //Garnier opera => 21h -> 01/02h
+ } elseif ($location == 'Garnier opera' && in_array($slot, ['Evening', 'After'])) {
+ //Check if evening
+ if ($slot == 'Evening') {
+ //Set begin at 21h
+ $session->setBegin(new \DateTime('21:00:00'));
+
+ //Set length at 5h
+ $session->setLength(new \DateTime('05:00:00'));
+
+ //Check if next day is premium
+ if ($premium) {
+ //Set length at 6h
+ $session->setLength(new \DateTime('06:00:00'));
+ }
+ //Check if after
+ } else {
+ //Set begin at 1h
+ $session->setBegin(new \DateTime('01:00:00'));
+
+ //Set length at 4h
+ $session->setLength(new \DateTime('04:00:00'));
+
+ //Check if next day is premium
+ if ($premium) {
+ //Set begin at 2h
+ $session->setBegin(new \DateTime('02:00:00'));
+
+ //Set length at 3h
+ $session->setLength(new \DateTime('03:00:00'));
+ }
+ }
+ //Trocadero esplanade|Tokyo palace|Swan island|Saint-Honore market|Orsay museum => 19h -> 01/02h
+ } elseif (in_array($location, ['Trocadero esplanade', 'Tokyo palace', 'Swan island', 'Saint-Honore market', 'Orsay museum']) && in_array($slot, ['Evening', 'After'])) {
+ //Check if evening
+ if ($slot == 'Evening') {
+ //Set begin at 19h
+ $session->setBegin(new \DateTime('19:00:00'));
+
+ //Check if next day is premium
+ if ($premium) {
+ //Set length at 7h
+ $session->setLength(new \DateTime('07:00:00'));
+ }
+ //Check if after
+ } else {
+ //Set begin at 1h
+ $session->setBegin(new \DateTime('01:00:00'));
+
+ //Set length at 4h
+ $session->setLength(new \DateTime('04:00:00'));
+
+ //Check if next day is premium
+ if ($premium) {
+ //Set begin at 2h
+ $session->setBegin(new \DateTime('02:00:00'));
+
+ //Set length at 3h
+ $session->setLength(new \DateTime('03:00:00'));
+ }
+ }
+ //Drawings' garden (Villette) => 14h -> 19h
+ } elseif ($location == 'Drawings\' garden' && $slot == 'Afternoon') {
+ //Set begin at 14h
+ $session->setBegin(new \DateTime('14:00:00'));
+
+ //Set length at 5h
+ $session->setLength(new \DateTime('05:00:00'));
+ //Colette place => 14h -> 21h
+ //TODO: add check here that it's a millegaux account ?
+ } elseif ($location == 'Colette place' && $slot == 'Afternoon') {
+ //Set begin at 14h
+ $session->setBegin(new \DateTime('14:00:00'));
+
+ //Set length at 7h
+ $session->setLength(new \DateTime('07:00:00'));
+ //Orleans gallery => 14h -> 18h
+ } elseif ($location == 'Orleans gallery' && $slot == 'Afternoon') {
+ //Set begin at 14h
+ $session->setBegin(new \DateTime('14:00:00'));
+
+ //Set length at 4h
+ $session->setLength(new \DateTime('04:00:00'));
+ //Monde garden => 14h -> 19h
+ //TODO: add check here that it's a raphael account ?
+ } elseif ($location == 'Monde garden' && $slot == 'Afternoon') {
+ //Set begin at 14h
+ $session->setBegin(new \DateTime('14:00:00'));
+
+ //Set length at 4h
+ $session->setLength(new \DateTime('05:00:00'));
+ //Combination not supported
+ //TODO: add Madeleine place|Bastille place|Vendome place ?
+ } else {
+ //Add error in flash message
+ $this->addFlash('error', $this->translator->trans('Session on %date% %location% %slot% not yet supported', ['%location%' => $this->translator->trans('at '.$data['location']), '%slot%' => $this->translator->trans('the '.strtolower(strval($data['slot']))), '%date%' => $data['date']->format('Y-m-d')]));
+
+ //Set title
+ $title = $this->translator->trans('Application add');
+
+ //Render the view
+ return $this->render('@RapsysAir/application/add.html.twig', ['title' => $title, 'form' => $form->createView()]+$this->context);
+ }
+
+ //Check if admin
+ if (!$this->checker->isGranted('ROLE_ADMIN') && $session->getStart() < new \DateTime('00:00:00')) {
+ //Add error in flash message
+ $this->addFlash('error', $this->translator->trans('Session in the past on %date% %location% %slot% not yet supported', ['%location%' => $this->translator->trans('at '.$data['location']), '%slot%' => $this->translator->trans('the '.strtolower(strval($data['slot']))), '%date%' => $data['date']->format('Y-m-d')]));
+
+ //Set title
+ $title = $this->translator->trans('Application add');
+
+ //Render the view
+ return $this->render('@RapsysAir/application/add.html.twig', ['title' => $title, 'form' => $form->createView()]+$this->context);
+ }
//Queue session save
- $manager->persist($session);
+ $this->manager->persist($session);
//Flush to get the ids
- #$manager->flush();
+ #$this->manager->flush();
- $this->addFlash('notice', $this->translator->trans('Session on %date% %location% %slot% created', ['%location%' => $this->translator->trans('at '.$data['location']), '%slot%' => $this->translator->trans('the '.strtolower($data['slot'])), '%date%' => $data['date']->format('Y-m-d')]));
+ $this->addFlash('notice', $this->translator->trans('Session on %date% %location% %slot% created', ['%location%' => $this->translator->trans('at '.$data['location']), '%slot%' => $this->translator->trans('the '.strtolower(strval($data['slot']))), '%date%' => $data['date']->format('Y-m-d')]));
}
//Set user
- $user = $this->getUser();
+ $user = $this->security->getUser();
//Replace with requested user for admin
- if ($this->isGranted('ROLE_ADMIN') && !empty($data['user'])) {
- $user = $this->getDoctrine()->getRepository(User::class)->findOneById($data['user']);
+ if ($this->checker->isGranted('ROLE_ADMIN') && !empty($data['user'])) {
+ $user = $this->doctrine->getRepository(User::class)->findOneById($data['user']);
}
//Protect application fetching
try {
//Retrieve application
- $application = $doctrine->getRepository(Application::class)->findOneBySessionUser($session, $user);
+ $application = $this->doctrine->getRepository(Application::class)->findOneBySessionUser($session, $user);
- //Add notice in flash message
- //TODO: set warning about application already exists bla bla bla...
- #$this->addFlash('notice', $this->translator->trans('Application request the %date% for %location% on the slot %slot% saved', ['%location%' => $data['location']->getTitle(), '%slot%' => $data['slot']->getTitle(), '%date%' => $data['date']->format('Y-m-d')]));
-
- //Add error message to mail field
- #$form->get('slot')->addError(new FormError($this->translator->trans('Application already exists')));
-
- //TODO: redirect anyway on uri with application highlighted
+ //Add warning in flash message
+ $this->addFlash('warning', $this->translator->trans('Application on %date% %location% %slot% already exists', ['%location%' => $this->translator->trans('at '.$data['location']), '%slot%' => $this->translator->trans('the '.strtolower(strval($data['slot']))), '%date%' => $data['date']->format('Y-m-d')]));
//Catch no application and session without identifier (not persisted&flushed) cases
- } catch (\Doctrine\ORM\NoResultException|\Doctrine\ORM\ORMInvalidArgumentException $e) {
+ } catch (NoResultException|ORMInvalidArgumentException $e) {
//Create the application
$application = new Application();
+ $application->setDance($data['dance']);
$application->setSession($session);
$application->setUser($user);
- $application->setCreated(new \DateTime('now'));
- $application->setUpdated(new \DateTime('now'));
//Refresh session updated field
$session->setUpdated(new \DateTime('now'));
//Queue session save
- $manager->persist($session);
+ $this->manager->persist($session);
//Queue application save
- $manager->persist($application);
+ $this->manager->persist($application);
//Flush to get the ids
- #$manager->flush();
+ $this->manager->flush();
//Add notice in flash message
- $this->addFlash('notice', $this->translator->trans('Application on %date% %location% %slot% created', ['%location%' => $this->translator->trans('at '.$data['location']), '%slot%' => $this->translator->trans('the '.strtolower($data['slot'])), '%date%' => $data['date']->format('Y-m-d')]));
+ $this->addFlash('notice', $this->translator->trans('Application on %date% %location% %slot% created', ['%location%' => $this->translator->trans('at '.$data['location']), '%slot%' => $this->translator->trans('the '.strtolower(strval($data['slot']))), '%date%' => $data['date']->format('Y-m-d')]));
}
-
- //Try unshort return field
- if (
- !empty($data['return']) &&
- ($unshort = $this->slugger->unshort($data['return'])) &&
- ($route = json_decode($unshort, true)) !== null
- ) {
- $return = $this->generateUrl($route['_route'], ['session' => $session->getId()?:1]+$route['_route_params']);
- }
-
- //XXX: Debug
- header('Content-Type: text/plain');
//Extract and process referer
if ($referer = $request->headers->get('referer')) {
//Try with referer path
try {
- var_dump($this->router);
- exit;
- var_dump($path = '/location');
- var_dump($this->router->match($path));
- var_dump($path = '/fr/emplacement');
- exit;
- var_dump($this->router->match());
- exit;
- var_dump($path);
- var_dump($query);
- exit;
+ //Save old context
+ $oldContext = $this->router->getContext();
+
+ //Force clean context
+ //XXX: prevent MethodNotAllowedException because current context method is POST in onevendor/symfony/routing/Matcher/Dumper/CompiledUrlMatcherTrait.php+42
+ $this->router->setContext(new RequestContext());
+
//Retrieve route matching path
$route = $this->router->match($path);
- var_dump($route);
- exit;
- //Verify that it differ from current one
- if (($name = $route['_route']) == $logout) {
- throw new ResourceNotFoundException('Identical referer and logout route');
- }
+ //Reset context
+ $this->router->setContext($oldContext);
+
+ //Clear old context
+ unset($oldContext);
+
+ //Extract name
+ $name = $route['_route'];
//Remove route and controller from route defaults
unset($route['_route'], $route['_controller']);
+ //Check if session view route
+ if ($name == 'rapsys_air_session_view' && !empty($route['id'])) {
+ //Replace id
+ $route['id'] = $session->getId();
+ //Other routes
+ } else {
+ //Set session
+ $route['session'] = $session->getId();
+ }
+
//Generate url
- $url = $this->router->generate($name, $route);
+ return $this->redirectToRoute($name, $route);
//No route matched
- } catch(ResourceNotFoundException $e) {
+ } catch (MethodNotAllowedException|ResourceNotFoundException $e) {
//Unset referer to fallback to default route
unset($referer);
}
}
- var_dump($request->headers->get('referer'));
- #var_dump($request->get('_route'));
- var_dump($return);
- exit;
-
- //Fetch slugger helper
- $slugger = $this->get('rapsys.slugger');
-
- var_dump($short = $slugger->short(json_encode(['_route' => $request->get('_route'), '_route_params' => $request->get('_route_params')])));
- $short[12] = 'T';
- var_dump($ret = json_decode($slugger->unshort($short), true));
- var_dump($ret);
- var_dump($this->generateUrl($ret['_route'], $ret['_route_params']));
- #var_dump(json_decode($slugger->unshort($data['return'])));
- #var_dump($application->getId());
- exit;
-
- //Init application
- $application = false;
-
- //Protect application fetching
- try {
- //TODO: handle admin case where we provide a user in extra
- $application = $doctrine->getRepository(Application::class)->findOneBySessionUser($session, $this->getUser());
-
- //Add error message to mail field
- $form->get('slot')->addError(new FormError($this->translator->trans('Application already exists')));
- //Catch no application cases
- //XXX: combine these catch when php 7.1 is available
- } catch (\Doctrine\ORM\NoResultException $e) {
- //Catch invalid argument because session is not already persisted
- } catch(\Doctrine\ORM\ORMInvalidArgumentException $e) {
- }
-
- //Create new application if none found
- if (!$application) {
- //Create the application
- $application = new Application();
- $application->setSession($session);
- //TODO: handle admin case where we provide a user in extra
- $application->setUser($this->getUser());
- $application->setCreated(new \DateTime('now'));
- $application->setUpdated(new \DateTime('now'));
- $manager->persist($application);
-
- //Flush to get the ids
- $manager->flush();
-
- //Add notice in flash message
- $this->addFlash('notice', $this->translator->trans('Application request the %date% for %location% on the slot %slot% saved', ['%location%' => $data['location']->getTitle(), '%slot%' => $data['slot']->getTitle(), '%date%' => $data['date']->format('Y-m-d')]));
-
- //Redirect to cleanup the form
- return $this->redirectToRoute('rapsys_air_admin');
- }
- }
-
- function test(Request $request) {
-
- //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('Monday this week + 5 week')
- );
-
- //Fetch sessions
- $sessions = $doctrine->getRepository(Session::class)->findAllByDatePeriod($period);
-
- //Init calendar
- $calendar = [];
-
- //Init month
- $month = null;
-
- //Iterate on each day
- foreach($period as $date) {
- //Init day in calendar
- $calendar[$Ymd = $date->format('Ymd')] = [
- 'title' => $date->format('d'),
- 'class' => [],
- 'sessions' => []
- ];
- //Append month for first day of month
- if ($month != $date->format('m')) {
- $month = $date->format('m');
- $calendar[$Ymd]['title'] .= '/'.$month;
- }
- //Deal with today
- if ($date->format('U') == ($today = strtotime('today'))) {
- $calendar[$Ymd]['title'] .= '/'.$month;
- $calendar[$Ymd]['current'] = true;
- $calendar[$Ymd]['class'][] = 'current';
- }
- //Disable passed days
- if ($date->format('U') < $today) {
- $calendar[$Ymd]['disabled'] = true;
- $calendar[$Ymd]['class'][] = 'disabled';
- }
- //Set next month days
- if ($date->format('m') > date('m')) {
- $calendar[$Ymd]['next'] = true;
- $calendar[$Ymd]['class'][] = 'next';
- }
- //Iterate on each session to find the one of the day
- foreach($sessions as $session) {
- if (($sessionYmd = $session->getDate()->format('Ymd')) == $Ymd) {
- //Count number of application
- $count = count($session->getApplications());
-
- //Compute classes
- $class = [];
- if ($session->getApplication()) {
- $class[] = 'granted';
- } elseif ($count == 0) {
- $class[] = 'orphaned';
- } elseif ($count > 1) {
- $class[] = 'disputed';
- } else {
- $class[] = 'pending';
- }
-
- //Add the session
- $calendar[$Ymd]['sessions'][$session->getSlot()->getId().$session->getLocation()->getId()] = [
- 'id' => $session->getId(),
- 'title' => ($count > 1?'['.$count.'] ':'').$session->getSlot()->getTitle().' '.$session->getLocation()->getTitle(),
- 'class' => $class
- ];
- }
- }
-
- //Sort sessions
- ksort($calendar[$Ymd]['sessions']);
- }
+ //Redirect to cleanup the form
+ return $this->redirectToRoute('rapsys_air', ['session' => $session->getId()]);
}
}