SonataAdminBundle: создание объекта из List View (часть 2)

Постановка задачи


В первой части статьи мы создали кнопку в строке List View писем, которая переводит нас на форму создания ответа. Однако остались нерешенными по крайней мере два важных вопроса:

  1. автоматическая привязка ответа к письму
  2. проверка прав пользователя на создание ответа


Автоматическая привязка ответа к письму


Существует как минимум три способа автоматической привязки ответа к письму на уровне SonataAdminBundle:
1) После создания формы перед сохранением сущности ответа в базу (для этого можно использовать метод prePersist объекта Admin). Главной особенностью этого способа является «непрозрачность» прикрепления одной сущности к другой для пользователя, работающего с формой, что может служить как достоинством, так и недостатком в зависимости от поставленных целей.
2) Во время создания формы путем настройки значения необходимого поля с помощью formBuilder. Этот подход может привести к нежелательному загромождению админ-класса кодом, реализующим логику формирования значения поля по умолчанию.
3) До создания формы. Для этого можно воспользоваться подходом, предлагаемым, например в [1], который заключается в переопределении родительского метода getNewInstance Admin-класса. Альтернативой такому решению может являться наследование от CRUD-контроллера (процесс наследования подробно описан в [2]) и определение в нем операции-зацепки [3] preCreate. Одним из преимуществ такой альтернативы является «штатная» возможность возвратить полноценный объект \Symfony\Component\HttpFoundation\Response () в случае, если это необходимо.
Какой из способов использовать, зависит от решаемой задачи, мы же рассмотрим третий с использованием наследования от CRUD-контроллера, поскольку, на наш взгляд, он является наименее очевидным с точки зрения реализации и наиболее гибким с точки зрения архитектуры.
В первую очередь, добавим третий аргумент в функцию admin.getRouteGenerator.generateUrl (). Это должен быть массив параметров запроса и мы добавим в него идентификатор письма, из строки которого создаем ответ. Для этого воспользуемся функцией admin.getUrlsafeIdentifier ()

{# src/AppBundle/Resources/views/CRUD/list__action_create_other_admin.html.twig #}

    
    Создать ответ



Таким образом, переход по ссылке позволит обратиться к createAction CRUD-контроллера, передав в качестве параметра запроса incoming_id идентификатор нужного нам письма. Теперь задача сводится к получению объекта письма по идентификатору и прикреплению к нему ответа. Мы ее будем решать путем определения пустого по умолчанию метода preCreate, который может возвращать объект \Symfony\Component\HttpFoundation\Response () или не возвращать ничего, просто модифицируя $object.

namespace Application\Sonata\AdminBundle\Controller;

use AppBundle\Entity\Response;
use Sonata\AdminBundle\Controller\CRUDController as BaseController;
use Symfony\Component\HttpFoundation\Request;

class CRUDController extends BaseController
{

    public function preCreate(Request $request, $object)
    {
        // Здесь Response - сущность нашего ответа, а не объект \Symfony\Component\HttpFoundation\Response()
        if ($object instanceof Response) {
            // Если передан идентификатор письма
            if ($incomingId = $request->get('incoming_id')) {
                // Если по идентификатору получен объект письма
                 if($incoming = $this->getDoctrine()->getRepository('AppBundle:Incoming')->find($incomingId)) {
                     // Прикрепление письма к ответу
                     $object->setIncoming($incoming);

                     // Или ответа к письму, оба варианта ПРИКРЕПЛЕНИЯ равнозначны
                     $incoming->setResponse($object);
                 }
            }
        }
    }

}


Главная задача, поставленная в начале статьи, а именно: создание сущности ответа из ListView писем — решена. Рассмотрим вопрос проверки прав пользователя на создание ответа.

Проверка прав пользователя на создание ответа


Непосредственно вопрос установки прав пользователя на создание ответа лежит вне темы данной статьи и может быть решен на основе подробной информации, изложенной в [4]. Мы же рассмотрим вопрос отображения или скрытия кнопки «Добавить ответ» в ListView ПИСЕМ в зависимости от наличия или отсутствия у пользователя прав на создание ОТВЕТА. Для этого нам снова придется внести изменения в файл list__action_create_other_admin.html.twig

{# src/AppBundle/Resources/views/CRUD/list__action_create_other_admin.html.twig #}
{% if template_variables.otherAdmin.securityHandler.isGranted(template_variables.otherAdmin, 'CREATE', template_variables.otherAdmin) and template_variables.otherAdmin.hasRoute('create') %}
 
     
     Создать ответ
 
{% endif %}


При этом мы проверяем возможность создания ответа не только на уровне системы безопасности, но и на уровне системы роутинга.

Резюме


В статье изложен вариант решения задачи по созданию в SonataAdminBundle некоторой сущности из ListView второй сущности, связанной с первой, с учетом наличия или отсутствия у пользователя соответствующих прав.

Ссылки на используемые ресурсы


  1. Populate resp. set default values on form resp. object or instance in SonataAdminBundle
  2. CREATING A CUSTOM ADMIN ACTION
  3. Операции-зацепки
  4. Security

© Habrahabr.ru