src/EventSubscriber/EasyAdminSubscriber.php line 57
<?php
# src/EventSubscriber/EasyAdminSubscriber.php
namespace App\EventSubscriber;
use App\Entity\ExportExcel;
use App\Entity\FrontTheme;
use App\Entity\Settings;
use App\Entity\User;
use Doctrine\Common\Collections\ArrayCollection;
use Doctrine\Common\Collections\Collection;
use Doctrine\Persistence\ManagerRegistry;
use EasyCorp\Bundle\EasyAdminBundle\Event\BeforeEntityDeletedEvent;
use EasyCorp\Bundle\EasyAdminBundle\Event\BeforeEntityPersistedEvent;
use EasyCorp\Bundle\EasyAdminBundle\Event\BeforeEntityUpdatedEvent;
use RecursiveDirectoryIterator;
use RecursiveIteratorIterator;
use ReflectionClass;
use ReflectionMethod;
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
use Symfony\Component\Security\Core\Security;
use Symfony\Component\PasswordHasher\Hasher\UserPasswordHasherInterface;
use Symfony\Component\Filesystem\Exception\IOExceptionInterface;
use Symfony\Component\Filesystem\Filesystem;
use Symfony\Component\Filesystem\Path;
use Symfony\Component\Finder\Finder;
use Symfony\Component\HttpFoundation\File\Exception\FileException;
use Symfony\Component\Yaml\Yaml;
use ZipArchive;
use Symfony\Contracts\Translation\TranslatorInterface;
class EasyAdminSubscriber implements EventSubscriberInterface
{
public function __construct(
private readonly Security $security,
private readonly UserPasswordHasherInterface $passwordEncoder,
private readonly ManagerRegistry $doctrine,
private readonly TranslatorInterface $translatorInterface
)
{
}
public static function getSubscribedEvents()
{
return [
BeforeEntityPersistedEvent::class => ['beforePersist'],
BeforeEntityUpdatedEvent::class => ['beforeUpdated'],
BeforeEntityDeletedEvent::class => ['beforeDelete']
];
}
public function beforePersist(BeforeEntityPersistedEvent $event)
{
$entity = $event->getEntityInstance();
if($entity instanceof FrontTheme){
$filesystem = new Filesystem();
$finder = new Finder();
$path = __DIR__."/../../public/uploads/".$entity->getName();
if(!$filesystem->exists($path) )
{
$filesystem->mkdir($path,0777);
$filesystem->mkdir($path."/pages",0777);
$filesystem->mkdir($path."/shop",0777);
}
$pathFrontBundle = __DIR__."/../Flexy/FrontBundle";
$themePath = $pathFrontBundle."/Themes";
$filesystem->mirror($themePath."/Default",$themePath."/".ucfirst($entity->getName()));
//$filesystem->mkdir(__DIR__."/../../public/themes/".strtolower($entity->getName()),0777);
$filesystem->symlink($themePath."/".$entity->getName()."/assets", __DIR__."/../../public/themes/".strtolower($entity->getName()));
//Step 1 : Copy Controllers
$finder->in($themePath."/".ucfirst($entity->getName())."/Controller");
$finder->files()->contains("Default");
foreach ($finder as $file) {
$file_contents = file_get_contents($file->getPathname());
$file_contents = str_replace("Default\Controller", ucfirst($entity->getName())."\Controller", $file_contents);
$file_contents = str_replace("Default\Form", ucfirst($entity->getName())."\Form", $file_contents);
file_put_contents($file->getPathname(), $file_contents);
}
//Step 2 : Copy Forms
$finder->in($themePath."/".ucfirst($entity->getName())."/Form");
$finder->files()->contains("Default");
//dd($finder->hasResults());
foreach ($finder as $file) {
$file_contents = file_get_contents($file->getPathname());
$file_contents = str_replace("Default\Form", ucfirst($entity->getName())."\Form", $file_contents);
file_put_contents($file->getPathname(), $file_contents);
}
//Step 3 : Copy assets
//$assetsPath = $pathFrontBundle."/Resources/public/assets";
//$filesystem->mirror($assetsPath."/default",$assetsPath."/".strtolower($entity->getName()));
//Step 4 : Copy Templates
$finder->in($themePath."/".ucfirst($entity->getName())."/templates");
$finder->files()->contains("default");
//dd($finder->hasResults());
foreach ($finder as $file) {
$file_contents = file_get_contents($file->getPathname());
$file_contents = str_replace("default/", strtolower($entity->getName())."/", $file_contents);
file_put_contents($file->getPathname(), $file_contents);
}
}
if($entity instanceof Settings){
if($entity->getFrontTheme()){
$entity->setAssetFolderName(strtolower($entity->getFrontTheme()->getName()));
}
$filesystem = new Filesystem();
$path = __DIR__."/../../public/uploads/".$entity->getAssetFolderName();
if(!$filesystem->exists($path) )
{
$filesystem->mkdir($path,0777);
$filesystem->mkdir($path."/pages",0777);
$filesystem->mkdir($path."/shop",0777);
}
$pathFrontBundleRoutes = __DIR__."/../Flexy/FrontBundle/Resources/config/routes.yaml";
if($filesystem->exists($pathFrontBundleRoutes)){
$configArray = Yaml::parseFile($pathFrontBundleRoutes);
$configArray["controllers"]["resource"] = "../../Controller/".ucfirst($entity->getAssetFolderName());
$yamlConfig = Yaml::dump($configArray);
file_put_contents($pathFrontBundleRoutes, $yamlConfig);
}
}
if ($entity instanceof ExportExcel) {
$entityNameSpace = $entity->getEntityNameSpace();
$directFields = $this->doctrine->getManager()->getClassMetadata($entityNameSpace)->getColumnNames();
$associationFields = $this->doctrine->getManager()->getClassMetadata($entityNameSpace)->getAssociationMappings();
$entityRepository = $this->doctrine->getRepository($entityNameSpace);
$repositoryMethods = $this->getAllRepositoryMethods($entityRepository);
$reflectionClass = $this->doctrine->getManager()->getClassMetadata($entityNameSpace)->reflClass->getMethods();// All methods detailled
//-------------
//dd($reflectionClass[0]->name);
$fields = [];
$translatedFields = [];
foreach($reflectionClass as $singleRefClass){
if(count($entity->getFields()) > 0 ){
if($entity->isExcludeFields()){
if(in_array($singleRefClass->name,$entity->getFields())){
continue;
}
}else{
if(!in_array($singleRefClass->name,$entity->getFields())){
continue;
}
}
}
if (substr($singleRefClass->name, 0, 3) === "get") {
$fields[]=$singleRefClass->name;
} else {
continue;
}
}
foreach($repositoryMethods as $singleMethod){
$fields[]=$singleMethod;
}
//$selectedFieldsToExport = ["name","categoryProduct","subCategoryProduct","price"];
$selectedFieldsToExport = [];
$selectedFieldsToExport = array_merge($selectedFieldsToExport,$fields);
//$selectedFieldsToExport = array_merge($selectedFieldsToExport,$reflectionClass);
if(count($entity->getFindBy())>0){
$queryBuilder = $this->doctrine->getManager()->createQueryBuilder();
$queryBuilder
->select('e')
->from($entityNameSpace, 'e')
//->join('e.relatedEntity', 'r')
->andWhere('JSON_CONTAINS(e.metaData, :metadataAPE) = 1 OR JSON_CONTAINS(e.metaData, :metadataNAF) = 1')
->setParameter("metadataAPE",'{"key": "CodeAPE"}')
->setParameter("metadataNAF",'{"key": "CodeNAF"}')
;
foreach($entity->getFindBy() as $key => $value){
$queryBuilder
->andWhere('e.'.$key.' '.$value);
}
if(count($entity->getOrderBy())>0){
foreach($entity->getOrderBy() as $key => $value){
if(gettype($key)=="integer"){
$queryBuilder->orderBy("e.".$value);
}else{
$queryBuilder->orderBy("e.".$key,$value);
}
}
}
//$queryBuilder->setMaxResults(500);
$entityFetchAll = $queryBuilder->getQuery()->getResult();
}else{
$queryBuilder = $this->doctrine->getManager()->createQueryBuilder();
$queryBuilder
->select('e')
->from($entityNameSpace, 'e');
//->join('e.relatedEntity', 'r')
$queryBuilder->setMaxResults(500);
$entityFetchAll = $queryBuilder->getQuery()->getResult();
}
$arrayToExport = [];
$correctedFieldsName =[];
foreach($fields as $field){
if($field == "getMetaData"){
continue;
}
$name = $this->getAbsoluteNameMethod($field);
$correctedFieldsName[]= $this->translatorInterface->trans($name,[],"admin");
}
foreach($entity->getMetaDataFields() as $singleField){
$correctedFieldsName[]= $this->translatorInterface->trans($singleField,[],"admin");
}
$arrayToExport[]=$correctedFieldsName;
foreach($entityFetchAll as $singleEntityInstance){
$singleRow =[];
$singleMetaDataValue = [];
foreach($fields as $singleField){
$value=null;
$valueSub=null;
$valueParent=null;
if(method_exists($singleEntityInstance,$singleField)){
$value = $singleEntityInstance->{$singleField}();
if($value instanceof Collection){
$value = implode("\n", $value->toArray());
//dd($value);
}elseif(is_array($value)){
if($singleField == "getMetaData"){
foreach($entity->getMetaDataFields() as $singleField){
$singleMetaDataValue[] = $this->getMetaDataValue($value,$singleField);
continue;
}
continue;
foreach($value as $k=>$v){
if(in_array($k,$entity->getMetaDataFields())){
dd($k);
$singleMetaDataValue[]=$v["value"];
}else
dd("None");
}
continue;
}
$value = $this->implode_multidimensional($value,"\n");
}elseif(gettype($value) == "string"){
$value = $this->translatorInterface->trans($value);
}
}elseif(method_exists($entityRepository,$singleField)){
$value = $entityRepository->{$singleField}($singleEntityInstance->getId());
if(is_array($value)){
// find the index of the value 'orange'
$field = $this->getAbsoluteNameMethod($singleField);
$translatedField = $this->translatorInterface->trans($field);
$index = array_search($translatedField, $arrayToExport[0]);
// if the value exists in the array, remove it
if ($index !== false) {
unset($arrayToExport[0][$index]);
}
// re-index the array
//$arrayToExport = array_values($arrayToExport[0]);
// dd($value);
foreach($value as $singleData){
$translatedKey = $this->translatorInterface->trans($singleData["key"]);
if(!in_array($translatedKey,$arrayToExport[0])){
$arrayToExport[0][] = $this->translatorInterface->trans($translatedKey);
}
$singleRow[]=$singleData["value"];
}
continue;
}else{
$value = $this->translatorInterface->trans($value);
}
}
$singleRow[]=$value;
}
foreach($singleMetaDataValue as $singleMetaDataVal){
//dd($singleMetaDataVal);
if(is_array($singleMetaDataVal)){
$singleRow[] = $this->implode_multidimensional($singleMetaDataVal,", ");
continue;
}
$singleRow[] = $singleMetaDataVal;
}
$arrayToExport[]=$singleRow;
//dd($arrayToExport);
}
//dd($arrayToExport);
$mySpreadsheet = new \PhpOffice\PhpSpreadsheet\Spreadsheet();
$mySpreadsheet->removeSheetByIndex(0);
$worksheet1 = new \PhpOffice\PhpSpreadsheet\Worksheet\Worksheet($mySpreadsheet, "List Produits");
$mySpreadsheet->addSheet($worksheet1, 0);
$worksheet1->fromArray($arrayToExport,null,"A1");
// Set the auto filter to cover all the data
$lastColumn = $worksheet1->getHighestColumn();
$lastRow = $worksheet1->getHighestDataRow();
$range = "A1:$lastColumn$lastRow";
$worksheet1->setAutoFilter($range);
foreach ($worksheet1->getColumnIterator() as $column)
{
$worksheet1->getColumnDimension($column->getColumnIndex())->setAutoSize(true);
}
foreach ($worksheet1->getRowIterator() as $row) {
foreach ($row->getCellIterator() as $cell) {
$cell->getStyle()->getAlignment()->setVertical(\PhpOffice\PhpSpreadsheet\Style\Alignment::VERTICAL_TOP);
}
}
$writer = new \PhpOffice\PhpSpreadsheet\Writer\Xlsx($mySpreadsheet);
$parts = explode("\\", $entityNameSpace);
$nameOfFile = end($parts);
$nameOfFile = strtolower($nameOfFile)."-".$entity->getCreatedAt()->format("Y-m-d-H-i");
//echo $lastWord; // output: CategoryProduct
$settings = $this->doctrine->getManager()->getRepository(Settings::class)->findOneBy(["code"=>"main"]);
$assetFolderName = $settings->getAssetFolderName();
$assetFolderName = strtolower($assetFolderName);
$writer->save("uploads/".$assetFolderName."/exports/".$nameOfFile.".xlsx");
$entity->setGeneratedFilePath("uploads/".$assetFolderName."/exports/".$nameOfFile.".xlsx");
$entity->setEntityNameSpace($entityNameSpace);
}
if ($entity instanceof User) {
$entity->setPassword(
$this->passwordEncoder->hashPassword(
$entity,
$entity->getPassword()
));
}
}
public function beforeUpdated(BeforeEntityUpdatedEvent $event)
{
$entity = $event->getEntityInstance();
if($entity instanceof Settings){
$filesystem = new Filesystem();
$pathFrontBundle = __DIR__."/../Flexy/FrontBundle";
$themePath = $pathFrontBundle."/Themes";
if($entity->getFrontTheme()){
$entity->setAssetFolderName($entity->getFrontTheme()->getName());
if(!$filesystem->exists(__DIR__."/../../public/themes/".strtolower($entity->getFrontTheme()->getName()))){
$filesystem->symlink($themePath."/".$entity->getFrontTheme()->getName()."/assets", __DIR__."/../../public/themes/".strtolower($entity->getFrontTheme()->getName()));
}
}
$path = __DIR__."/../../public/uploads/".$entity->getAssetFolderName();
if(!$filesystem->exists($path) )
{
$filesystem->mkdir($path,0777);
$filesystem->mkdir($path."/pages",0777);
$filesystem->mkdir($path."/shop",0777);
}
$pathFrontBundleRoutes = __DIR__."/../Flexy/FrontBundle/Resources/config/routes.yaml";
if($filesystem->exists($pathFrontBundleRoutes)){
$configArray = Yaml::parseFile($pathFrontBundleRoutes);
$configArray["controllers"]["resource"] = "../../Themes/".ucfirst($entity->getAssetFolderName())."/Controller";
$yamlConfig = Yaml::dump($configArray);
file_put_contents($pathFrontBundleRoutes, $yamlConfig);
}
}
}
public function beforeDelete(BeforeEntityDeletedEvent $event){
$entity = $event->getEntityInstance();
if($entity instanceof FrontTheme){
//$entity->getSettings()->setFrontTheme(null);
$filesystem = new Filesystem();
//dd(__DIR__."/../../");
$pathFrontBundle = __DIR__."/../Flexy/FrontBundle";
$themePath = $pathFrontBundle."/Themes";
if($filesystem->exists($themePath."/".ucfirst($entity->getName()) ) ){
$this->zipDirectory($themePath."/".ucfirst($entity->getName()),__DIR__."/../../public/archive-themes/".strtolower($entity->getName()).".zip" );
$filesystem->remove($themePath."/".ucfirst($entity->getName()));
}
//$filesystem->remove($pathFrontBundle."/Resources/public/assets/".strtolower($entity->getName()));
$settings = $this->doctrine->getManager()->getRepository(Settings::class)->findOneBy(["code"=>"main"]);
$currentTheme = $settings->getAssetFolderName();
$entity->removeSetting($settings);
$settings->setAssetFolderName("Default");
$this->doctrine->getManager()->persist($settings);
if(strtolower($entity->getName()) == strtolower($currentTheme) ){
$pathFrontBundleRoutes = __DIR__."/../Flexy/FrontBundle/Resources/config/routes.yaml";
if($filesystem->exists($pathFrontBundleRoutes)){
$configArray = Yaml::parseFile($pathFrontBundleRoutes);
$configArray["controllers"]["resource"] = "../../Themes/Default/Controller";
$yamlConfig = Yaml::dump($configArray);
file_put_contents($pathFrontBundleRoutes, $yamlConfig);
}
}
}
}
public function exportFile($records) {
$heading = false;
if(!empty($records))
foreach($records as $row) {
if(!$heading) {
// display field/column names as a first row
echo implode("\t", array_keys($row)) . "\n";
$heading = true;
}
echo implode("\t", array_values($row)) . "\n";
}
}
public static function zipDirectory($directory, $zipName)
{
// Get real path for our folder
$rootPath = realpath($directory);
// Initialize archive object
$zip = new ZipArchive();
$zip->open($zipName, ZipArchive::CREATE | ZipArchive::OVERWRITE);
// Initialize empty "delete list"
$filesToDelete = array();
// Create recursive directory iterator
/** @var SplFileInfo[] $files */
$files = new RecursiveIteratorIterator(
new RecursiveDirectoryIterator($rootPath),
RecursiveIteratorIterator::LEAVES_ONLY
);
foreach ($files as $name => $file)
{
// Skip directories (they would be added automatically)
if (!$file->isDir())
{
// Get real and relative path for current file
$filePath = $file->getRealPath();
$relativePath = substr($filePath, strlen($rootPath) + 1);
// Add current file to archive
$zip->addFile($filePath, $relativePath);
// Add current file to "delete list"
// delete it later cause ZipArchive create archive only after calling close function and ZipArchive lock files until archive created)
if ($file->getFilename() != 'important.txt')
{
$filesToDelete[] = $filePath;
}
}
}
// Zip archive will be created only after closing object
$zip->close();
return $zip;
}
function getAbsoluteNameMethod($input) {
// Remove the "get" prefix from the input string
$input = preg_replace('/^get/', '', $input);
// Replace all capital letters with a space followed by the letter
$output = preg_replace('/([A-Z])/', ' $1', $input);
// Remove any leading or trailing spaces
$output = trim($output);
// Convert the first character to uppercase
$output = ucfirst($output);
// Return the final result
return $output;
}
public function getAllRepositoryMethods($repository)
{
$reflectionClass = new ReflectionClass(get_class($repository));
$methods = $reflectionClass->getMethods(ReflectionMethod::IS_PUBLIC);
$findMethods = [];
foreach ($methods as $method) {
$methodName = $method->getName();
if (strpos($methodName, 'export') === 0) {
$findMethods[] = $methodName;
}
}
return $findMethods;
}
function is_multidimensional(array $array) {
foreach ($array as $element) {
if (is_array($element)) {
return true;
}
}
return false;
}
function implode_multidimensional(array $array, $glue = ', ') {
$result = [];
foreach ($array as $key => $element) {
if (is_array($element)) {
$element = $this->implode_multidimensional($element, $glue);
}
//$result[] = "{$key}:{$element}";
if(!in_array("{$element}",$result)){
$result[] = "{$element}";
}
}
return implode($glue, $result);
}
public function getMetaDataValue($jsonData, $searchKey)
{
$dataArray = (array)$jsonData;
foreach ($dataArray as $item) {
if (isset($item['key']) && $item['key'] === $searchKey) {
return $item['value'];
}
}
return null;
}
}