|
<?php |
|
|
|
declare(strict_types=1); |
|
|
|
/* |
|
* (c) Alexandre Rock Ancelet <[email protected]> and Studio Agate. |
|
* |
|
* Licensed with MIT |
|
*/ |
|
|
|
namespace App\Admin\Controller; |
|
|
|
use Admin\DTO\EasyAdminDTOInterface; |
|
use EasyCorp\Bundle\EasyAdminBundle\Controller\EasyAdminController; |
|
use Symfony\Component\Form\FormInterface; |
|
|
|
trait BaseDTOControllerTrait |
|
{ |
|
abstract protected function getDTOClass(): string; |
|
|
|
/** |
|
* @return object an instance of the configured Entity |
|
*/ |
|
abstract protected function createEntityFromDTO(EasyAdminDTOInterface $dto): object; |
|
|
|
/** |
|
* You can return a new instance if you want to override any existing entity instead of updating one (i.e when you have immutable objects). |
|
* |
|
* @return object an instance of the configured Entity, or the same object that is passed as argument |
|
*/ |
|
abstract protected function updateEntityWithDTO(object $entity, EasyAdminDTOInterface $dto): object; |
|
|
|
protected function createNewEntity() |
|
{ |
|
$dtoClass = $this->doGetDTOClass(); |
|
|
|
return $dtoClass::createEmpty(); |
|
} |
|
|
|
protected function persistEntity($object): void |
|
{ |
|
$dtoClass = $this->doGetDTOClass(); |
|
|
|
if (!($object instanceof $dtoClass)) { |
|
throw new \InvalidArgumentException('DTO is not of valid type.'); |
|
} |
|
|
|
parent::persistEntity($this->createEntityFromDTO($object)); |
|
} |
|
|
|
protected function getEntityFormOptions($entity, $view) |
|
{ |
|
$dtoClass = $this->doGetDTOClass(); |
|
|
|
$options = parent::getEntityFormOptions($entity, $view); |
|
$options['data_class'] = $dtoClass; |
|
|
|
return $options; |
|
} |
|
|
|
protected function createEntityFormBuilder($entity, $view) |
|
{ |
|
$dtoClass = $this->doGetDTOClass(); |
|
|
|
if (!($entity instanceof $dtoClass)) { |
|
$entity = $dtoClass::createFromEntity($entity); |
|
} |
|
|
|
return parent::createEntityFormBuilder($entity, $view); |
|
} |
|
|
|
protected function updateEntity($entity, FormInterface $editForm = null): void |
|
{ |
|
if (!$editForm) { |
|
throw new \InvalidArgumentException('Form is mandatory to update entity.'); |
|
} |
|
|
|
$dtoClass = $this->doGetDTOClass(); |
|
$entityClass = $this->entity['class']; |
|
$dto = $editForm->getData(); |
|
|
|
if (!($dto instanceof $dtoClass)) { |
|
throw new \InvalidArgumentException(\sprintf('DTO is not of valid type, expected %s.', $dtoClass)); |
|
} |
|
|
|
if (!($entity instanceof $entityClass)) { |
|
throw new \InvalidArgumentException(\sprintf('Only %s is supported in this controller.', $entityClass)); |
|
} |
|
|
|
parent::updateEntity($this->updateEntityWithDTO($entity, $dto)); |
|
} |
|
|
|
private function doGetDTOClass(): string |
|
{ |
|
$this->validateDTOClass(); |
|
|
|
return $this->getDTOClass(); |
|
} |
|
|
|
private function validateDTOClass(): void |
|
{ |
|
if (!($this instanceof EasyAdminController)) { |
|
throw new \RuntimeException(\sprintf('The DTO-based controller %s must extend %s.', \get_class($this), EasyAdminController::class)); |
|
} |
|
|
|
$dtoClass = $this->getDTOClass(); |
|
if (!\is_subclass_of($dtoClass, EasyAdminDTOInterface::class, true)) { |
|
throw new \RuntimeException(\sprintf('DTO class %s must implement %s.', $dtoClass, EasyAdminDTOInterface::class)); |
|
} |
|
} |
|
} |