From 1717ce5c2c48abb02d7d4c42f0b21f2c0d881b42 Mon Sep 17 00:00:00 2001 From: Simon Vieille Date: Tue, 5 Apr 2022 12:05:06 +0200 Subject: [PATCH] add projects --- config/packages/app.yaml | 2 + src/Controller/ProjectAdminController.php | 149 ++++++++++++++++++ src/Controller/ProjectController.php | 17 ++ src/Entity/Project.php | 117 ++++++++++++++ src/Factory/ProjectFactory.php | 14 ++ src/Form/ProjectLinkType.php | 50 ++++++ src/Form/ProjectType.php | 82 ++++++++++ src/Repository/ProjectRepository.php | 76 +++++++++ src/Repository/ProjectRepositoryQuery.php | 29 ++++ templates/admin/menu.html.twig | 13 +- .../blog/project_admin/field/status.html.twig | 7 + .../blog/project_admin/field/title.html.twig | 20 +++ templates/page/simple/projects.html.twig | 36 +++++ 13 files changed, 610 insertions(+), 2 deletions(-) create mode 100644 src/Controller/ProjectAdminController.php create mode 100644 src/Controller/ProjectController.php create mode 100644 src/Entity/Project.php create mode 100644 src/Factory/ProjectFactory.php create mode 100644 src/Form/ProjectLinkType.php create mode 100644 src/Form/ProjectType.php create mode 100644 src/Repository/ProjectRepository.php create mode 100644 src/Repository/ProjectRepositoryQuery.php create mode 100644 templates/blog/project_admin/field/status.html.twig create mode 100644 templates/blog/project_admin/field/title.html.twig create mode 100644 templates/page/simple/projects.html.twig diff --git a/config/packages/app.yaml b/config/packages/app.yaml index 77db520..8b0d1ad 100644 --- a/config/packages/app.yaml +++ b/config/packages/app.yaml @@ -18,11 +18,13 @@ core: - {name: 'Blog\PostFollowController::enable', action: 'App\Controller\Blog\PostFollowController::enable'} - {name: 'Blog\PostFollowController::disable', action: 'App\Controller\Blog\PostFollowController::disable'} - {name: 'StlMeshController::meshes', action: 'App\Controller\StlMeshController::meshes'} + - {name: 'ProjectController::projects', action: 'App\Controller\ProjectController::projects'} pages: App\Entity\Page\SimplePage: name: 'Page de contenu' templates: - {name: "Par défaut", file: "page/simple/default.html.twig"} + - {name: "Projets", file: "page/simple/projects.html.twig"} App\Entity\Page\PostPage: name: 'Page article' templates: diff --git a/src/Controller/ProjectAdminController.php b/src/Controller/ProjectAdminController.php new file mode 100644 index 0000000..f889f20 --- /dev/null +++ b/src/Controller/ProjectAdminController.php @@ -0,0 +1,149 @@ +doIndex($page, $query, $request, $session); + } + + /** + * @Route("/admin/project/new", name="admin_project_new", methods={"GET", "POST"}) + */ + public function new(Factory $factory, EntityManager $entityManager, Request $request): Response + { + return $this->doNew($factory->create(), $entityManager, $request); + } + + /** + * @Route("/admin/project/show/{entity}", name="admin_project_show", methods={"GET"}) + */ + public function show(Entity $entity): Response + { + return $this->doShow($entity); + } + + /** + * @Route("/admin/project/filter", name="admin_project_filter", methods={"GET"}) + */ + public function filter(Session $session): Response + { + return $this->doFilter($session); + } + + /** + * @Route("/admin/project/edit/{entity}", name="admin_project_edit", methods={"GET", "POST"}) + */ + public function edit(Entity $entity, EntityManager $entityManager, Request $request): Response + { + return $this->doEdit($entity, $entityManager, $request); + } + + /** + * @Route("/admin/project/sort/{page}", name="admin_project_sort", methods={"POST"}, requirements={"page":"\d+"}) + */ + public function sort(RepositoryQuery $query, EntityManager $entityManager, Request $request, Session $session, int $page = 1): Response + { + return $this->doSort($page, $query, $entityManager, $request, $session); + } + + /** + * @Route("/admin/project/batch/{page}", name="admin_project_batch", methods={"POST"}, requirements={"page":"\d+"}) + */ + public function batch(RepositoryQuery $query, EntityManager $entityManager, Request $request, Session $session, int $page = 1): Response + { + return $this->doBatch($page, $query, $entityManager, $request, $session); + } + + /** + * @Route("/admin/project/delete/{entity}", name="admin_project_delete", methods={"DELETE"}) + */ + public function delete(Entity $entity, EntityManager $entityManager, Request $request): Response + { + return $this->doDelete($entity, $entityManager, $request); + } + + protected function getConfiguration(): CrudConfiguration + { + return CrudConfiguration::create() + ->setPageTitle('index', 'Projets') + ->setPageTitle('edit', '{label}') + ->setPageTitle('new', 'Nouveau projet') + // ->setPageTitle('show', 'View of {id}') + + ->setPageRoute('index', 'admin_project_index') + ->setPageRoute('new', 'admin_project_new') + ->setPageRoute('edit', 'admin_project_edit') + // ->setPageRoute('show', 'admin_project_show') + ->setPageRoute('sort', 'admin_project_sort') + ->setPageRoute('batch', 'admin_project_batch') + ->setPageRoute('delete', 'admin_project_delete') + ->setPageRoute('filter', 'admin_project_filter') + + ->setForm('edit', Type::class, []) + ->setForm('new', Type::class) + // ->setForm('filter', Type::class) + + // ->setMaxPerPage('index', 20) + + ->setIsSortableCollection('index', true) + // ->setSortableCollectionProperty('sortOrder') + + // ->setAction('index', 'new', true) + ->setAction('index', 'show', false) + // ->setAction('index', 'edit', true) + // ->setAction('index', 'delete', true) + + // ->setAction('edit', 'back', true) + ->setAction('edit', 'show', false) + // ->setAction('edit', 'delete', true) + + // ->setAction('show', 'back', true) + // ->setAction('show', 'edit', true) + + ->setField('index', 'Label', Field\TextField::class, [ + 'property' => 'label', + ]) + ->setField('index', 'Status', Field\TextField::class, [ + 'view' => 'blog/project_admin/field/status.html.twig', + 'sort' => ['status', '.status'], + 'attr' => ['class' => 'miw-100'], + ]) + + ->setBatchAction('index', 'delete', 'Delete', function(EntityInterface $entity, EntityManager $manager) { + $manager->delete($entity); + }) + ->setBatchAction('index', 'draft', 'Statut : publier', function(EntityInterface $entity, EntityManager $manager) { + $manager->update($entity->setStatus(Project::PUBLISHED)); + }) + ->setBatchAction('index', 'publish', 'Statut : brouillon', function(EntityInterface $entity, EntityManager $manager) { + $manager->update($entity->setStatus(Project::DRAFT)); + }) + ; + } + + protected function getSection(): string + { + return 'project'; + } +} diff --git a/src/Controller/ProjectController.php b/src/Controller/ProjectController.php new file mode 100644 index 0000000..f181c7b --- /dev/null +++ b/src/Controller/ProjectController.php @@ -0,0 +1,17 @@ +defaultRender($this->siteRequest->getPage()->getTemplate(), [ + 'projects' => $query->create()->published()->ordered()->find(), + ]); + } +} diff --git a/src/Entity/Project.php b/src/Entity/Project.php new file mode 100644 index 0000000..da69d2b --- /dev/null +++ b/src/Entity/Project.php @@ -0,0 +1,117 @@ +id; + } + + public function getLabel(): ?string + { + return $this->label; + } + + public function setLabel(string $label): self + { + $this->label = $label; + + return $this; + } + + public function getDescription(): ?string + { + return $this->description; + } + + public function setDescription(?string $description): self + { + $this->description = $description; + + return $this; + } + + public function getStatus(): ?int + { + return $this->status; + } + + public function setStatus(int $status): self + { + $this->status = $status; + + return $this; + } + + public function getSortOrder(): ?int + { + return $this->sortOrder; + } + + public function setSortOrder(?int $sortOrder): self + { + $this->sortOrder = $sortOrder; + + return $this; + } + + public function getLinks(): ?array + { + return $this->links; + } + + public function setLinks(array $links): self + { + $this->links = $links; + + return $this; + } +} diff --git a/src/Factory/ProjectFactory.php b/src/Factory/ProjectFactory.php new file mode 100644 index 0000000..cd20c00 --- /dev/null +++ b/src/Factory/ProjectFactory.php @@ -0,0 +1,14 @@ +add( + 'label', + TextType::class, + [ + 'label' => 'Libellé', + 'required' => true, + 'attr' => [ + ], + 'constraints' => [ + new NotBlank(), + ], + ] + ); + + $builder->add( + 'url', + TextType::class, + [ + 'label' => 'URL', + 'required' => true, + 'attr' => [ + ], + 'constraints' => [ + new NotBlank(), + ], + ] + ); + } + + public function configureOptions(OptionsResolver $resolver): void + { + $resolver->setDefaults([ + // Configure your form options here + ]); + } +} diff --git a/src/Form/ProjectType.php b/src/Form/ProjectType.php new file mode 100644 index 0000000..02ea1cf --- /dev/null +++ b/src/Form/ProjectType.php @@ -0,0 +1,82 @@ +add( + 'label', + TextType::class, + [ + 'label' => 'Libellé', + 'required' => true, + 'attr' => [ + ], + 'constraints' => [ + new NotBlank(), + ], + ] + ); + + $builder->add( + 'status', + ChoiceType::class, + [ + 'label' => 'Statut', + 'required' => true, + 'choices' => [ + 'Brouillon' => Project::DRAFT, + 'Publié' => Project::PUBLISHED, + ], + 'attr' => [ + ], + 'constraints' => [ + new NotBlank(), + ], + ] + ); + + $builder->add( + 'description', + SimpleMdTextareaType::class, + [ + 'label' => 'Contenu', + 'required' => false, + 'constraints' => [ + ], + ] + ); + + $builder->add( + 'links', + CollectionType::class, + [ + 'label' => 'Liens', + 'entry_type' => ProjectLinkType::class, + 'by_reference' => false, + 'allow_add' => true, + 'allow_delete' => true, + 'prototype' => true, + ] + ); + } + + public function configureOptions(OptionsResolver $resolver): void + { + $resolver->setDefaults([ + 'data_class' => Project::class, + ]); + } +} diff --git a/src/Repository/ProjectRepository.php b/src/Repository/ProjectRepository.php new file mode 100644 index 0000000..560cb87 --- /dev/null +++ b/src/Repository/ProjectRepository.php @@ -0,0 +1,76 @@ +_em->persist($entity); + if ($flush) { + $this->_em->flush(); + } + } + + /** + * @throws ORMException + * @throws OptimisticLockException + */ + public function remove(Project $entity, bool $flush = true): void + { + $this->_em->remove($entity); + if ($flush) { + $this->_em->flush(); + } + } + + // /** + // * @return Project[] Returns an array of Project objects + // */ + /* + public function findByExampleField($value) + { + return $this->createQueryBuilder('p') + ->andWhere('p.exampleField = :val') + ->setParameter('val', $value) + ->orderBy('p.id', 'ASC') + ->setMaxResults(10) + ->getQuery() + ->getResult() + ; + } + */ + + /* + public function findOneBySomeField($value): ?Project + { + return $this->createQueryBuilder('p') + ->andWhere('p.exampleField = :val') + ->setParameter('val', $value) + ->getQuery() + ->getOneOrNullResult() + ; + } + */ +} diff --git a/src/Repository/ProjectRepositoryQuery.php b/src/Repository/ProjectRepositoryQuery.php new file mode 100644 index 0000000..b4fa818 --- /dev/null +++ b/src/Repository/ProjectRepositoryQuery.php @@ -0,0 +1,29 @@ +andWhere('.status = :published') + ->setParameter(':published', Project::PUBLISHED) + ; + } + + public function ordered() + { + return $this->orderBy('.sortOrder'); + } +} diff --git a/templates/admin/menu.html.twig b/templates/admin/menu.html.twig index 1f7b846..48e2acb 100644 --- a/templates/admin/menu.html.twig +++ b/templates/admin/menu.html.twig @@ -25,12 +25,21 @@ }) }} -{{ include('@Core/admin/module/_menu_section.html.twig', {label: 'Impression 3D'}) }} +{{ include('@Core/admin/module/_menu_section.html.twig', {label: 'Projets'}) }} + +