* * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Sonata\AdminBundle\Controller; use Symfony\Component\HttpFoundation\RedirectResponse; use Symfony\Component\HttpFoundation\Response; use Symfony\Component\HttpKernel\Exception\NotFoundHttpException; use Symfony\Component\Security\Core\Exception\AccessDeniedException; use Symfony\Component\DependencyInjection\ContainerInterface; use Symfony\Bundle\FrameworkBundle\Controller\Controller; class CRUDController extends Controller { /** * The related Admin class * * @var \Sonata\AdminBundle\Admin\AdminInterface */ protected $admin; /** * @param mixed $data * @param integer $status * @param array $headers * * @return Response with json encoded data */ public function renderJson($data, $status = 200, $headers = array()) { // fake content-type so browser does not show the download popup when this // response is rendered through an iframe (used by the jquery.form.js plugin) // => don't know yet if it is the best solution if ($this->get('request')->get('_xml_http_request') && strpos($this->get('request')->headers->get('Content-Type'), 'multipart/form-data') === 0) { $headers['Content-Type'] = 'text/plain'; } else { $headers['Content-Type'] = 'application/json'; } return new Response(json_encode($data), $status, $headers); } /** * * @return boolean true if the request is done by an ajax like query */ public function isXmlHttpRequest() { return $this->get('request')->isXmlHttpRequest() || $this->get('request')->get('_xml_http_request'); } /** * Sets the Container associated with this Controller. * * @param ContainerInterface $container A ContainerInterface instance */ public function setContainer(ContainerInterface $container = null) { $this->container = $container; $this->configure(); } /** * Contextualize the admin class depends on the current request * * @throws \RuntimeException * @return void */ public function configure() { $adminCode = $this->container->get('request')->get('_sonata_admin'); if (!$adminCode) { throw new \RuntimeException(sprintf('There is no `_sonata_admin` defined for the controller `%s` and the current route `%s`', get_class($this), $this->container->get('request')->get('_route'))); } $this->admin = $this->container->get('sonata.admin.pool')->getAdminByAdminCode($adminCode); if (!$this->admin) { throw new \RuntimeException(sprintf('Unable to find the admin class related to the current controller (%s)', get_class($this))); } $rootAdmin = $this->admin; if ($this->admin->isChild()) { $this->admin->setCurrentChild(true); $rootAdmin = $rootAdmin->getParent(); } $rootAdmin->setRequest($this->container->get('request')); } /** * return the base template name * * @return string the template name */ public function getBaseTemplate() { if ($this->isXmlHttpRequest()) { return $this->container->getParameter('sonata.admin.templates.ajax'); } return $this->container->getParameter('sonata.admin.templates.layout'); } /** * @param $view * @param array $parameters * @param null|\Symfony\Component\HttpFoundation\Response $response * @return \Symfony\Bundle\FrameworkBundle\Controller\Response */ public function render($view, array $parameters = array(), Response $response = null) { $parameters['admin'] = isset($parameters['admin']) ? $parameters['admin'] : $this->admin; $parameters['base_template'] = isset($parameters['base_template']) ? $parameters['base_template'] : $this->getBaseTemplate(); return parent::render($view, $parameters); } /** * return the Response object associated to the list action * * @return Response */ public function listAction() { if (false === $this->admin->isGranted('LIST')) { throw new AccessDeniedException(); } return $this->render($this->admin->getListTemplate(), array( 'action' => 'list', )); } /** * execute a batch delete * * @param array $idx * @return \Symfony\Component\HttpFoundation\RedirectResponse */ public function batchActionDelete($query) { if (false === $this->admin->isGranted('DELETE')) { throw new AccessDeniedException(); } $modelManager = $this->admin->getModelManager(); $modelManager->batchDelete($this->admin->getClass(), $query); $this->get('session')->setFlash('sonata_flash_success', 'flash_batch_delete_success'); return new RedirectResponse($this->admin->generateUrl('list', $this->admin->getFilterParameters())); } public function deleteAction($id) { if (false === $this->admin->isGranted('DELETE')) { throw new AccessDeniedException(); } $id = $this->get('request')->get($this->admin->getIdParameter()); $object = $this->admin->getObject($id); if (!$object) { throw new NotFoundHttpException(sprintf('unable to find the object with id : %s', $id)); } $this->admin->delete($object); $this->get('session')->setFlash('sonata_flash_success', 'flash_delete_success'); return new RedirectResponse($this->admin->generateUrl('list')); } /** * return the Response object associated to the edit action * * @throws \Symfony\Component\HttpKernel\Exception\NotFoundHttpException * @param $id * @return \Symfony\Component\HttpFoundation\Response */ public function editAction($id) { if (false === $this->admin->isGranted('EDIT')) { throw new AccessDeniedException(); } $object = $this->admin->getObject($this->get('request')->get($this->admin->getIdParameter())); if (!$object) { throw new NotFoundHttpException(sprintf('unable to find the object with id : %s', $id)); } $this->admin->setSubject($object); $form = $this->admin->getForm(); $form->setData($object); if ($this->get('request')->getMethod() == 'POST') { $form->bindRequest($this->get('request')); if ($form->isValid()) { $this->admin->update($object); $this->get('session')->setFlash('sonata_flash_success', 'flash_edit_success'); if ($this->isXmlHttpRequest()) { return $this->renderJson(array( 'result' => 'ok', 'objectId' => $this->admin->getNormalizedIdentifier($object) )); } // redirect to edit mode return $this->redirectTo($object); } $this->get('session')->setFlash('sonata_flash_error', 'flash_edit_error'); } $view = $form->createView(); // set the theme for the current Admin Form $this->get('twig')->getExtension('form')->setTheme($view, $this->admin->getFormTheme()); return $this->render($this->admin->getEditTemplate(), array( 'action' => 'edit', 'form' => $view, 'object' => $object, )); } /** * redirect the user depend on this choice * * @param $object * @return \Symfony\Component\HttpFoundation\Response */ public function redirectTo($object) { $url = false; if ($this->get('request')->get('btn_update_and_list')) { $url = $this->admin->generateUrl('list'); } if ($this->get('request')->get('btn_create_and_create')) { $url = $this->admin->generateUrl('create'); } if (!$url) { $url = $this->admin->generateObjectUrl('edit', $object); } return new RedirectResponse($url); } /** * return the Response object associated to the batch action * * @throws \RuntimeException * @return \Symfony\Component\HttpFoundation\Response */ public function batchAction() { if ($this->get('request')->getMethod() != 'POST') { throw new \RuntimeException('invalid request type, POST expected'); } $action = $this->get('request')->get('action'); $idx = $this->get('request')->get('idx'); $all_elements = $this->get('request')->get('all_elements', false); if (count($idx) == 0 && !$all_elements) { // no item selected $this->get('session')->setFlash('sonata_flash_notice', 'flash_batch_empty'); return new RedirectResponse($this->admin->generateUrl('list', $this->admin->getFilterParameters())); } if (!array_key_exists($action, $this->admin->getBatchActions())) { throw new \RuntimeException(sprintf('The `%s` batch action is not defined', $action)); } // execute the action, batchActionXxxxx $action = \Sonata\AdminBundle\Admin\BaseFieldDescription::camelize($action); $final_action = sprintf('batchAction%s', ucfirst($action)); if (!method_exists($this, $final_action)) { throw new \RuntimeException(sprintf('A `%s::%s` method must be created', get_class($this), $final_action)); } $datagrid = $this->admin->getDatagrid(); $datagrid->buildPager(); $query = $datagrid->getQuery(); $query->setFirstResult(null); $query->setMaxResults(null); if (count($idx) > 0) { $this->admin->getModelManager()->addIdentifiersToQuery($this->admin->getClass(), $query, $idx); } return call_user_func(array($this, $final_action), $query); } /** * return the Response object associated to the create action * * @return \Symfony\Component\HttpFoundation\Response */ public function createAction() { if (false === $this->admin->isGranted('CREATE')) { throw new AccessDeniedException(); } $object = $this->admin->getNewInstance(); $form = $this->admin->getForm(); $form->setData($object); $this->admin->setSubject($object); if ($this->get('request')->getMethod() == 'POST') { $form->bindRequest($this->get('request')); if ($form->isValid()) { $this->admin->create($object); if ($this->isXmlHttpRequest()) { return $this->renderJson(array( 'result' => 'ok', 'objectId' => $this->admin->getNormalizedIdentifier($object) )); } $this->get('session')->setFlash('sonata_flash_success','flash_create_success'); // redirect to edit mode return $this->redirectTo($object); } $this->get('session')->setFlash('sonata_flash_error', 'flash_create_error'); } $view = $form->createView(); // set the theme for the current Admin Form $this->get('twig')->getExtension('form')->setTheme($view, $this->admin->getFormTheme()); return $this->render($this->admin->getEditTemplate(), array( 'action' => 'create', 'form' => $view, 'object' => $object, )); } /** * return the Response object associated to the view action * * @return \Symfony\Component\HttpFoundation\Response */ public function showAction($id) { if (false === $this->admin->isGranted('SHOW')) { throw new AccessDeniedException(); } $object = $this->admin->getObject($this->get('request')->get($this->admin->getIdParameter())); if (!$object) { throw new NotFoundHttpException(sprintf('unable to find the object with id : %s', $id)); } $this->admin->setSubject($object); // build the show list $elements = $this->admin->getShow(); return $this->render($this->admin->getShowTemplate(), array( 'action' => 'show', 'object' => $object, 'elements' => $this->admin->getShow(), )); } }