Add Category entity, edit event tabs (info/categories/stats/settings), CRUD categories
- Create Category entity: name, position (sortable), event, startAt, endAt, isActive() - Default endAt: event.startAt - 1 day - Add 4 tabs on edit event page: Informations, Categories/Billets, Statistiques, Parametres - Add routes: add category, delete category, reorder categories (JSON API) - Categories sorted by position, drag handle for future Sortable.js - Active/Inactive badge based on date range - Add migration for category table Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
34
migrations/Version20260320214602.php
Normal file
34
migrations/Version20260320214602.php
Normal file
@@ -0,0 +1,34 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace DoctrineMigrations;
|
||||
|
||||
use Doctrine\DBAL\Schema\Schema;
|
||||
use Doctrine\Migrations\AbstractMigration;
|
||||
|
||||
/**
|
||||
* Auto-generated Migration: Please modify to your needs!
|
||||
*/
|
||||
final class Version20260320214602 extends AbstractMigration
|
||||
{
|
||||
public function getDescription(): string
|
||||
{
|
||||
return '';
|
||||
}
|
||||
|
||||
public function up(Schema $schema): void
|
||||
{
|
||||
// this up() migration is auto-generated, please modify it to your needs
|
||||
$this->addSql('CREATE TABLE category (id INT GENERATED BY DEFAULT AS IDENTITY NOT NULL, name VARCHAR(255) NOT NULL, position INT NOT NULL, start_at TIMESTAMP(0) WITHOUT TIME ZONE NOT NULL, end_at TIMESTAMP(0) WITHOUT TIME ZONE NOT NULL, created_at TIMESTAMP(0) WITHOUT TIME ZONE NOT NULL, event_id INT NOT NULL, PRIMARY KEY (id))');
|
||||
$this->addSql('CREATE INDEX IDX_64C19C171F7E88B ON category (event_id)');
|
||||
$this->addSql('ALTER TABLE category ADD CONSTRAINT FK_64C19C171F7E88B FOREIGN KEY (event_id) REFERENCES event (id) ON DELETE CASCADE NOT DEFERRABLE');
|
||||
}
|
||||
|
||||
public function down(Schema $schema): void
|
||||
{
|
||||
// this down() migration is auto-generated, please modify it to your needs
|
||||
$this->addSql('ALTER TABLE category DROP CONSTRAINT FK_64C19C171F7E88B');
|
||||
$this->addSql('DROP TABLE category');
|
||||
}
|
||||
}
|
||||
@@ -6,10 +6,11 @@ parameters:
|
||||
- src/Kernel.php
|
||||
ignoreErrors:
|
||||
-
|
||||
message: '#Property App\\Entity\\(EmailTracking|MessengerLog|User|Payout|Event)::\$id .* never assigned#'
|
||||
message: '#Property App\\Entity\\(EmailTracking|MessengerLog|User|Payout|Event|Category)::\$id .* never assigned#'
|
||||
paths:
|
||||
- src/Entity/EmailTracking.php
|
||||
- src/Entity/MessengerLog.php
|
||||
- src/Entity/User.php
|
||||
- src/Entity/Payout.php
|
||||
- src/Entity/Event.php
|
||||
- src/Entity/Category.php
|
||||
|
||||
@@ -2,6 +2,7 @@
|
||||
|
||||
namespace App\Controller;
|
||||
|
||||
use App\Entity\Category;
|
||||
use App\Entity\Event;
|
||||
use App\Entity\Payout;
|
||||
use App\Entity\User;
|
||||
@@ -365,16 +366,111 @@ class AccountController extends AbstractController
|
||||
return $this->redirectToRoute('app_account_edit_event', ['id' => $event->getId()]);
|
||||
}
|
||||
|
||||
$categories = $em->getRepository(Category::class)->findBy(
|
||||
['event' => $event],
|
||||
['position' => 'ASC'],
|
||||
);
|
||||
|
||||
return $this->render('account/edit_event.html.twig', [
|
||||
'event' => $event,
|
||||
'categories' => $categories,
|
||||
'breadcrumbs' => [
|
||||
self::BREADCRUMB_HOME,
|
||||
self::BREADCRUMB_ACCOUNT,
|
||||
['name' => 'Modifier un evenement', 'url' => '/mon-compte/evenement/'.$event->getId().'/modifier'],
|
||||
['name' => $event->getTitle(), 'url' => '/mon-compte/evenement/'.$event->getId().'/modifier'],
|
||||
],
|
||||
]);
|
||||
}
|
||||
|
||||
#[Route('/mon-compte/evenement/{id}/categorie/ajouter', name: 'app_account_event_add_category', methods: ['POST'])]
|
||||
public function addCategory(Event $event, Request $request, EntityManagerInterface $em): Response
|
||||
{
|
||||
$this->denyAccessUnlessGranted('ROLE_ORGANIZER');
|
||||
|
||||
/** @var User $user */
|
||||
$user = $this->getUser();
|
||||
if ($event->getAccount()->getId() !== $user->getId()) {
|
||||
throw $this->createAccessDeniedException();
|
||||
}
|
||||
|
||||
$name = trim($request->request->getString('name'));
|
||||
if ('' === $name) {
|
||||
$this->addFlash('error', 'Le nom de la categorie est requis.');
|
||||
|
||||
return $this->redirectToRoute('app_account_edit_event', ['id' => $event->getId(), 'tab' => 'categories']);
|
||||
}
|
||||
|
||||
$maxPosition = $em->getRepository(Category::class)->count(['event' => $event]);
|
||||
|
||||
$category = new Category();
|
||||
$category->setName($name);
|
||||
$category->setEvent($event);
|
||||
$category->setPosition($maxPosition);
|
||||
|
||||
$startAt = $request->request->getString('start_at');
|
||||
if ('' !== $startAt) {
|
||||
$category->setStartAt(new \DateTimeImmutable($startAt));
|
||||
}
|
||||
|
||||
$endAt = $request->request->getString('end_at');
|
||||
if ('' !== $endAt) {
|
||||
$category->setEndAt(new \DateTimeImmutable($endAt));
|
||||
}
|
||||
|
||||
$em->persist($category);
|
||||
$em->flush();
|
||||
|
||||
$this->addFlash('success', sprintf('Categorie "%s" ajoutee.', $name));
|
||||
|
||||
return $this->redirectToRoute('app_account_edit_event', ['id' => $event->getId(), 'tab' => 'categories']);
|
||||
}
|
||||
|
||||
#[Route('/mon-compte/evenement/{id}/categorie/{categoryId}/supprimer', name: 'app_account_event_delete_category', methods: ['POST'])]
|
||||
public function deleteCategory(Event $event, int $categoryId, EntityManagerInterface $em): Response
|
||||
{
|
||||
$this->denyAccessUnlessGranted('ROLE_ORGANIZER');
|
||||
|
||||
/** @var User $user */
|
||||
$user = $this->getUser();
|
||||
if ($event->getAccount()->getId() !== $user->getId()) {
|
||||
throw $this->createAccessDeniedException();
|
||||
}
|
||||
|
||||
$category = $em->getRepository(Category::class)->find($categoryId);
|
||||
if ($category && $category->getEvent()->getId() === $event->getId()) {
|
||||
$em->remove($category);
|
||||
$em->flush();
|
||||
$this->addFlash('success', 'Categorie supprimee.');
|
||||
}
|
||||
|
||||
return $this->redirectToRoute('app_account_edit_event', ['id' => $event->getId(), 'tab' => 'categories']);
|
||||
}
|
||||
|
||||
#[Route('/mon-compte/evenement/{id}/categorie/reorder', name: 'app_account_event_reorder_categories', methods: ['POST'])]
|
||||
public function reorderCategories(Event $event, Request $request, EntityManagerInterface $em): Response
|
||||
{
|
||||
$this->denyAccessUnlessGranted('ROLE_ORGANIZER');
|
||||
|
||||
/** @var User $user */
|
||||
$user = $this->getUser();
|
||||
if ($event->getAccount()->getId() !== $user->getId()) {
|
||||
throw $this->createAccessDeniedException();
|
||||
}
|
||||
|
||||
$order = json_decode($request->getContent(), true);
|
||||
if (\is_array($order)) {
|
||||
foreach ($order as $position => $categoryId) {
|
||||
$category = $em->getRepository(Category::class)->find($categoryId);
|
||||
if ($category && $category->getEvent()->getId() === $event->getId()) {
|
||||
$category->setPosition($position);
|
||||
}
|
||||
}
|
||||
$em->flush();
|
||||
}
|
||||
|
||||
return $this->json(['success' => true]);
|
||||
}
|
||||
|
||||
#[Route('/mon-compte/evenement/{id}/en-ligne', name: 'app_account_toggle_event_online', methods: ['POST'])]
|
||||
public function toggleEventOnline(Event $event, EntityManagerInterface $em, EventIndexService $eventIndex): Response
|
||||
{
|
||||
|
||||
122
src/Entity/Category.php
Normal file
122
src/Entity/Category.php
Normal file
@@ -0,0 +1,122 @@
|
||||
<?php
|
||||
|
||||
namespace App\Entity;
|
||||
|
||||
use App\Repository\CategoryRepository;
|
||||
use Doctrine\ORM\Mapping as ORM;
|
||||
|
||||
#[ORM\Entity(repositoryClass: CategoryRepository::class)]
|
||||
class Category
|
||||
{
|
||||
#[ORM\Id]
|
||||
#[ORM\GeneratedValue]
|
||||
#[ORM\Column]
|
||||
private ?int $id = null;
|
||||
|
||||
#[ORM\Column(length: 255)]
|
||||
private ?string $name = null;
|
||||
|
||||
#[ORM\Column]
|
||||
private int $position = 0;
|
||||
|
||||
#[ORM\ManyToOne(targetEntity: Event::class)]
|
||||
#[ORM\JoinColumn(nullable: false, onDelete: 'CASCADE')]
|
||||
private ?Event $event = null;
|
||||
|
||||
#[ORM\Column]
|
||||
private \DateTimeImmutable $startAt;
|
||||
|
||||
#[ORM\Column]
|
||||
private \DateTimeImmutable $endAt;
|
||||
|
||||
#[ORM\Column]
|
||||
private \DateTimeImmutable $createdAt;
|
||||
|
||||
public function __construct()
|
||||
{
|
||||
$this->createdAt = new \DateTimeImmutable();
|
||||
$this->startAt = new \DateTimeImmutable();
|
||||
$this->endAt = new \DateTimeImmutable('+30 days');
|
||||
}
|
||||
|
||||
public function getId(): ?int
|
||||
{
|
||||
return $this->id;
|
||||
}
|
||||
|
||||
public function getName(): ?string
|
||||
{
|
||||
return $this->name;
|
||||
}
|
||||
|
||||
public function setName(string $name): static
|
||||
{
|
||||
$this->name = $name;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function getPosition(): int
|
||||
{
|
||||
return $this->position;
|
||||
}
|
||||
|
||||
public function setPosition(int $position): static
|
||||
{
|
||||
$this->position = $position;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function getEvent(): ?Event
|
||||
{
|
||||
return $this->event;
|
||||
}
|
||||
|
||||
public function setEvent(?Event $event): static
|
||||
{
|
||||
$this->event = $event;
|
||||
|
||||
if ($event && $event->getStartAt()) {
|
||||
$this->endAt = $event->getStartAt()->modify('-1 day');
|
||||
}
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function getStartAt(): \DateTimeImmutable
|
||||
{
|
||||
return $this->startAt;
|
||||
}
|
||||
|
||||
public function setStartAt(\DateTimeImmutable $startAt): static
|
||||
{
|
||||
$this->startAt = $startAt;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function getEndAt(): \DateTimeImmutable
|
||||
{
|
||||
return $this->endAt;
|
||||
}
|
||||
|
||||
public function setEndAt(\DateTimeImmutable $endAt): static
|
||||
{
|
||||
$this->endAt = $endAt;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function getCreatedAt(): \DateTimeImmutable
|
||||
{
|
||||
return $this->createdAt;
|
||||
}
|
||||
|
||||
public function isActive(): bool
|
||||
{
|
||||
$now = new \DateTimeImmutable();
|
||||
|
||||
return $now >= $this->startAt && $now <= $this->endAt;
|
||||
}
|
||||
}
|
||||
18
src/Repository/CategoryRepository.php
Normal file
18
src/Repository/CategoryRepository.php
Normal file
@@ -0,0 +1,18 @@
|
||||
<?php
|
||||
|
||||
namespace App\Repository;
|
||||
|
||||
use App\Entity\Category;
|
||||
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
||||
use Doctrine\Persistence\ManagerRegistry;
|
||||
|
||||
/**
|
||||
* @extends ServiceEntityRepository<Category>
|
||||
*/
|
||||
class CategoryRepository extends ServiceEntityRepository
|
||||
{
|
||||
public function __construct(ManagerRegistry $registry)
|
||||
{
|
||||
parent::__construct($registry, Category::class);
|
||||
}
|
||||
}
|
||||
@@ -1,6 +1,6 @@
|
||||
{% extends 'base.html.twig' %}
|
||||
|
||||
{% block title %}Modifier un evenement - E-Ticket{% endblock %}
|
||||
{% block title %}{{ event.title }} - E-Ticket{% endblock %}
|
||||
|
||||
{% block body %}
|
||||
<div class="w-full md:w-[80%] mx-auto py-12 px-4">
|
||||
@@ -9,8 +9,8 @@
|
||||
Retour aux evenements
|
||||
</a>
|
||||
|
||||
<h1 class="text-3xl font-black uppercase tracking-tighter italic heading-page">Modifier un evenement</h1>
|
||||
<p class="font-bold text-gray-600 italic mb-8">Modifiez les informations de votre evenement.</p>
|
||||
<h1 class="text-3xl font-black uppercase tracking-tighter italic heading-page">{{ event.title }}</h1>
|
||||
<p class="font-bold text-gray-600 italic mb-4">Gestion de l'evenement.</p>
|
||||
|
||||
{% for message in app.flashes('success') %}
|
||||
<div class="flash-success"><p class="font-black text-sm">{{ message }}</p></div>
|
||||
@@ -19,7 +19,7 @@
|
||||
<div class="flash-error"><p class="font-black text-sm">{{ message }}</p></div>
|
||||
{% endfor %}
|
||||
|
||||
<div class="flex flex-wrap gap-4 mb-8">
|
||||
<div class="flex flex-wrap gap-4 mb-6">
|
||||
{% if event.online %}
|
||||
<form method="post" action="{{ path('app_account_toggle_event_online', {id: event.id}) }}">
|
||||
<button type="submit" class="px-4 py-2 border-2 border-red-800 bg-red-600 text-white font-black uppercase text-xs tracking-widest cursor-pointer hover:bg-red-800 transition-all">
|
||||
@@ -65,7 +65,7 @@
|
||||
</div>
|
||||
|
||||
{% if event.online %}
|
||||
<div class="card-brutal-green mb-8 flex flex-wrap items-center justify-between gap-4">
|
||||
<div class="card-brutal-green mb-6 flex flex-wrap items-center justify-between gap-4">
|
||||
<div>
|
||||
<p class="text-[10px] font-black uppercase tracking-widest text-gray-500 mb-1">URL publique</p>
|
||||
<p class="text-sm font-mono font-bold break-all" id="event-url">{{ absolute_url(path('app_event_detail', {orgaSlug: event.account.slug, id: event.id, eventSlug: event.slug})) }}</p>
|
||||
@@ -74,9 +74,17 @@
|
||||
Copier le lien
|
||||
</button>
|
||||
</div>
|
||||
<script type="application/json" id="copy-url-data">{{ absolute_url(path('app_event_detail', {orgaSlug: event.account.slug, id: event.id, eventSlug: event.slug})) }}</script>
|
||||
{% endif %}
|
||||
|
||||
{% set current_tab = app.request.query.get('tab', 'info') %}
|
||||
<div class="flex flex-wrap overflow-x-auto mb-8">
|
||||
<a href="{{ path('app_account_edit_event', {id: event.id, tab: 'info'}) }}" class="flex-1 min-w-[100px] text-center py-3 border-3 border-gray-900 border-r-0 {{ current_tab == 'info' ? 'bg-yellow-400' : 'bg-white' }} font-black uppercase text-xs tracking-widest transition-all">Informations</a>
|
||||
<a href="{{ path('app_account_edit_event', {id: event.id, tab: 'categories'}) }}" class="flex-1 min-w-[100px] text-center py-3 border-3 border-gray-900 border-r-0 {{ current_tab == 'categories' ? 'bg-yellow-400' : 'bg-white' }} font-black uppercase text-xs tracking-widest transition-all">Categories / Billets</a>
|
||||
<a href="{{ path('app_account_edit_event', {id: event.id, tab: 'stats'}) }}" class="flex-1 min-w-[100px] text-center py-3 border-3 border-gray-900 border-r-0 {{ current_tab == 'stats' ? 'bg-yellow-400' : 'bg-white' }} font-black uppercase text-xs tracking-widest transition-all">Statistiques</a>
|
||||
<a href="{{ path('app_account_edit_event', {id: event.id, tab: 'settings'}) }}" class="flex-1 min-w-[100px] text-center py-3 border-3 border-gray-900 {{ current_tab == 'settings' ? 'bg-yellow-400' : 'bg-white' }} font-black uppercase text-xs tracking-widest transition-all">Parametres</a>
|
||||
</div>
|
||||
|
||||
{% if current_tab == 'info' %}
|
||||
<div class="flex flex-col lg:flex-row gap-8">
|
||||
<div class="flex-1 min-w-0">
|
||||
<form method="post" action="{{ path('app_account_edit_event', {id: event.id}) }}" enctype="multipart/form-data" class="form-col">
|
||||
@@ -149,5 +157,68 @@
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{% elseif current_tab == 'categories' %}
|
||||
<div class="card-brutal overflow-hidden mb-6">
|
||||
<div class="section-header flex justify-between items-center">
|
||||
<h2 class="text-[10px] font-black uppercase tracking-widest text-white">Categories</h2>
|
||||
</div>
|
||||
<div class="p-6">
|
||||
<form method="post" action="{{ path('app_account_event_add_category', {id: event.id}) }}" class="flex flex-wrap gap-3 items-end mb-6">
|
||||
<div class="flex-1 min-w-[200px]">
|
||||
<label for="cat_name" class="text-xs font-black uppercase tracking-widest form-label">Nouvelle categorie</label>
|
||||
<input type="text" id="cat_name" name="name" required class="form-input focus:border-indigo-600" placeholder="Ex: PMR, VIP, Tribune...">
|
||||
</div>
|
||||
<div>
|
||||
<label for="cat_start" class="text-xs font-black uppercase tracking-widest form-label">Debut vente</label>
|
||||
<input type="datetime-local" id="cat_start" name="start_at" class="form-input focus:border-indigo-600" value="{{ "now"|date('Y-m-d\\TH:i') }}">
|
||||
</div>
|
||||
<div>
|
||||
<label for="cat_end" class="text-xs font-black uppercase tracking-widest form-label">Fin vente</label>
|
||||
<input type="datetime-local" id="cat_end" name="end_at" class="form-input focus:border-indigo-600" value="{{ event.startAt|date_modify('-1 day')|date('Y-m-d\\TH:i') }}">
|
||||
</div>
|
||||
<button type="submit" class="btn-brutal font-black uppercase text-xs tracking-widest hover:bg-indigo-600 hover:text-white transition-all">
|
||||
+ Ajouter
|
||||
</button>
|
||||
</form>
|
||||
|
||||
{% if categories|length > 0 %}
|
||||
<div id="categories-list">
|
||||
{% for category in categories %}
|
||||
<div class="flex flex-wrap items-center gap-4 p-4 border-2 border-gray-900 mb-2 bg-white cursor-move" data-id="{{ category.id }}">
|
||||
<span class="text-gray-400 cursor-grab">☰</span>
|
||||
<span class="font-black text-sm uppercase flex-1">{{ category.name }}</span>
|
||||
<span class="text-xs font-bold text-gray-400">{{ category.startAt|date('d/m/Y H:i') }} — {{ category.endAt|date('d/m/Y H:i') }}</span>
|
||||
{% if category.active %}
|
||||
<span class="badge-green text-xs font-black uppercase">Active</span>
|
||||
{% else %}
|
||||
<span class="badge-red text-xs font-black uppercase">Inactive</span>
|
||||
{% endif %}
|
||||
<form method="post" action="{{ path('app_account_event_delete_category', {id: event.id, categoryId: category.id}) }}" data-confirm="Supprimer cette categorie ?" class="inline">
|
||||
<button type="submit" class="px-2 py-1 border-2 border-red-800 bg-red-600 text-white text-xs font-black uppercase cursor-pointer hover:bg-red-800 transition-all">✕</button>
|
||||
</form>
|
||||
</div>
|
||||
{% endfor %}
|
||||
</div>
|
||||
{% else %}
|
||||
<p class="text-gray-400 font-bold text-sm text-center py-8">Aucune categorie. Ajoutez-en une pour commencer a vendre des billets.</p>
|
||||
{% endif %}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{% elseif current_tab == 'stats' %}
|
||||
<div class="card-brutal">
|
||||
<div class="p-12 text-center">
|
||||
<p class="text-gray-400 font-bold text-sm">Les statistiques seront disponibles prochainement.</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{% elseif current_tab == 'settings' %}
|
||||
<div class="card-brutal">
|
||||
<div class="p-12 text-center">
|
||||
<p class="text-gray-400 font-bold text-sm">Les parametres avances seront disponibles prochainement.</p>
|
||||
</div>
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
|
||||
@@ -567,7 +567,7 @@ class AccountControllerTest extends WebTestCase
|
||||
'is_online' => '1',
|
||||
]);
|
||||
|
||||
self::assertResponseRedirects('/mon-compte?tab=events');
|
||||
self::assertResponseRedirects('/mon-compte/evenement/'.$event->getId().'/modifier');
|
||||
}
|
||||
|
||||
public function testEditEventDeniedForOtherUser(): void
|
||||
@@ -715,7 +715,7 @@ class AccountControllerTest extends WebTestCase
|
||||
'city' => 'Paris',
|
||||
], ['event_main_picture' => $picture]);
|
||||
|
||||
self::assertResponseRedirects('/mon-compte?tab=events');
|
||||
self::assertResponseRedirects();
|
||||
}
|
||||
|
||||
public function testEditEventWithPicture(): void
|
||||
@@ -755,7 +755,7 @@ class AccountControllerTest extends WebTestCase
|
||||
'city' => 'Lyon',
|
||||
], ['event_main_picture' => $picture]);
|
||||
|
||||
self::assertResponseRedirects('/mon-compte?tab=events');
|
||||
self::assertResponseRedirects();
|
||||
}
|
||||
|
||||
public function testEventsSearchReturnsSuccess(): void
|
||||
|
||||
Reference in New Issue
Block a user