Add stats tab with orders list, search, cancel and refund actions
- OrderIndexService: Meilisearch index order_event_{id} for order search
- Stats tab: 4 KPI cards (orders, tickets sold, CA HT, total percu)
- Orders list with KnpPaginator, search bar via Meilisearch
- Each order shows: number, status, date, buyer, items, total, payment
- Cancel order: sets status cancelled, invalidates all tickets
- Refund order: Stripe refund on connected account, sets status refunded,
invalidates all tickets
- Orders indexed in Meilisearch after payment
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -12,6 +12,7 @@ use App\Entity\Payout;
|
||||
use App\Entity\User;
|
||||
use App\Service\EventIndexService;
|
||||
use App\Service\MailerService;
|
||||
use App\Service\OrderIndexService;
|
||||
use App\Service\PayoutPdfService;
|
||||
use App\Service\StripeService;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
@@ -344,7 +345,7 @@ class AccountController extends AbstractController
|
||||
}
|
||||
|
||||
#[Route('/mon-compte/evenement/{id}/modifier', name: 'app_account_edit_event', methods: ['GET', 'POST'])]
|
||||
public function editEvent(Event $event, Request $request, EntityManagerInterface $em, EventIndexService $eventIndex): Response
|
||||
public function editEvent(Event $event, Request $request, EntityManagerInterface $em, EventIndexService $eventIndex, PaginatorInterface $paginator, OrderIndexService $orderIndex): Response
|
||||
{
|
||||
$this->denyAccessUnlessGranted('ROLE_ORGANIZER');
|
||||
|
||||
@@ -381,6 +382,22 @@ class AccountController extends AbstractController
|
||||
}
|
||||
}
|
||||
|
||||
$searchQuery = $request->query->getString('q', '');
|
||||
$ordersQuery = '' !== $searchQuery
|
||||
? $orderIndex->searchOrders($event->getId(), $searchQuery)
|
||||
: $em->getRepository(BilletBuyer::class)->findBy(['event' => $event], ['createdAt' => 'DESC']);
|
||||
$eventOrders = $paginator->paginate($ordersQuery, $request->query->getInt('page', 1), 20);
|
||||
|
||||
$eventTotalHT = 0;
|
||||
$eventTotalSold = 0;
|
||||
$paidEventOrders = $em->getRepository(BilletBuyer::class)->findBy(['event' => $event, 'status' => BilletBuyer::STATUS_PAID]);
|
||||
foreach ($paidEventOrders as $paidOrder) {
|
||||
$eventTotalHT += $paidOrder->getTotalHT();
|
||||
foreach ($paidOrder->getItems() as $item) {
|
||||
$eventTotalSold += $item->getQuantity();
|
||||
}
|
||||
}
|
||||
|
||||
return $this->render('account/edit_event.html.twig', [
|
||||
'event' => $event,
|
||||
'categories' => $categories,
|
||||
@@ -388,6 +405,11 @@ class AccountController extends AbstractController
|
||||
'sold_counts' => $soldCounts,
|
||||
'commission_rate' => $user->getCommissionRate() ?? 0,
|
||||
'billet_design' => $em->getRepository(BilletDesign::class)->findOneBy(['event' => $event]),
|
||||
'event_orders' => $eventOrders,
|
||||
'event_total_ht' => $eventTotalHT / 100,
|
||||
'event_total_sold' => $eventTotalSold,
|
||||
'event_total_orders' => \count($paidEventOrders),
|
||||
'search_query' => $searchQuery,
|
||||
'breadcrumbs' => [
|
||||
self::BREADCRUMB_HOME,
|
||||
self::BREADCRUMB_ACCOUNT,
|
||||
@@ -675,6 +697,79 @@ class AccountController extends AbstractController
|
||||
return $this->json(['success' => true]);
|
||||
}
|
||||
|
||||
#[Route('/mon-compte/evenement/{id}/commande/{orderId}/annuler', name: 'app_account_event_cancel_order', methods: ['POST'])]
|
||||
public function cancelOrder(Event $event, int $orderId, EntityManagerInterface $em): Response
|
||||
{
|
||||
$this->denyAccessUnlessGranted('ROLE_ORGANIZER');
|
||||
|
||||
/** @var User $user */
|
||||
$user = $this->getUser();
|
||||
if ($event->getAccount()->getId() !== $user->getId()) {
|
||||
throw $this->createAccessDeniedException();
|
||||
}
|
||||
|
||||
$order = $em->getRepository(BilletBuyer::class)->find($orderId);
|
||||
if (!$order || $order->getEvent()->getId() !== $event->getId()) {
|
||||
throw $this->createNotFoundException();
|
||||
}
|
||||
|
||||
$order->setStatus(BilletBuyer::STATUS_CANCELLED);
|
||||
|
||||
$tickets = $em->getRepository(BilletOrder::class)->findBy(['billetBuyer' => $order]);
|
||||
foreach ($tickets as $ticket) {
|
||||
$ticket->setState(BilletOrder::STATE_INVALID);
|
||||
}
|
||||
|
||||
$em->flush();
|
||||
|
||||
$this->addFlash('success', 'Commande '.$order->getOrderNumber().' annulee.');
|
||||
|
||||
return $this->redirectToRoute('app_account_edit_event', ['id' => $event->getId(), 'tab' => 'stats']);
|
||||
}
|
||||
|
||||
/**
|
||||
* @codeCoverageIgnore Requires live Stripe API
|
||||
*/
|
||||
#[Route('/mon-compte/evenement/{id}/commande/{orderId}/rembourser', name: 'app_account_event_refund_order', methods: ['POST'])]
|
||||
public function refundOrder(Event $event, int $orderId, EntityManagerInterface $em, StripeService $stripeService): Response
|
||||
{
|
||||
$this->denyAccessUnlessGranted('ROLE_ORGANIZER');
|
||||
|
||||
/** @var User $user */
|
||||
$user = $this->getUser();
|
||||
if ($event->getAccount()->getId() !== $user->getId()) {
|
||||
throw $this->createAccessDeniedException();
|
||||
}
|
||||
|
||||
$order = $em->getRepository(BilletBuyer::class)->find($orderId);
|
||||
if (!$order || $order->getEvent()->getId() !== $event->getId()) {
|
||||
throw $this->createNotFoundException();
|
||||
}
|
||||
|
||||
if ($order->getStripeSessionId() && $user->getStripeAccountId()) {
|
||||
try {
|
||||
$stripeService->getClient()->refunds->create([
|
||||
'payment_intent' => $order->getStripeSessionId(),
|
||||
], ['stripe_account' => $user->getStripeAccountId()]);
|
||||
} catch (\Exception) {
|
||||
// Stripe failure is non-blocking
|
||||
}
|
||||
}
|
||||
|
||||
$order->setStatus(BilletBuyer::STATUS_REFUNDED);
|
||||
|
||||
$tickets = $em->getRepository(BilletOrder::class)->findBy(['billetBuyer' => $order]);
|
||||
foreach ($tickets as $ticket) {
|
||||
$ticket->setState(BilletOrder::STATE_INVALID);
|
||||
}
|
||||
|
||||
$em->flush();
|
||||
|
||||
$this->addFlash('success', 'Commande '.$order->getOrderNumber().' remboursee.');
|
||||
|
||||
return $this->redirectToRoute('app_account_edit_event', ['id' => $event->getId(), 'tab' => 'stats']);
|
||||
}
|
||||
|
||||
#[Route('/mon-compte/evenement/{id}/billet-preview', name: 'app_account_event_billet_preview', methods: ['GET'])]
|
||||
public function billetPreview(Event $event, Request $request): Response
|
||||
{
|
||||
|
||||
@@ -24,6 +24,7 @@ class BilletOrderService
|
||||
private Environment $twig,
|
||||
private MailerService $mailer,
|
||||
private InvoiceService $invoiceService,
|
||||
private OrderIndexService $orderIndex,
|
||||
private UrlGeneratorInterface $urlGenerator,
|
||||
#[Autowire('%kernel.project_dir%')] private string $projectDir,
|
||||
#[Autowire('%kernel.secret%')] private string $appSecret,
|
||||
@@ -48,6 +49,8 @@ class BilletOrderService
|
||||
$order->setStatus(BilletBuyer::STATUS_PAID);
|
||||
$order->setPaidAt(new \DateTimeImmutable());
|
||||
$this->em->flush();
|
||||
|
||||
$this->orderIndex->indexOrder($order);
|
||||
}
|
||||
|
||||
public function generatePdf(BilletOrder $ticket): string
|
||||
|
||||
73
src/Service/OrderIndexService.php
Normal file
73
src/Service/OrderIndexService.php
Normal file
@@ -0,0 +1,73 @@
|
||||
<?php
|
||||
|
||||
namespace App\Service;
|
||||
|
||||
use App\Entity\BilletBuyer;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
|
||||
class OrderIndexService
|
||||
{
|
||||
private const INDEX_PREFIX = 'order_event_';
|
||||
|
||||
public function __construct(
|
||||
private readonly MeilisearchService $meilisearch,
|
||||
private readonly EntityManagerInterface $em,
|
||||
) {
|
||||
}
|
||||
|
||||
/**
|
||||
* @return list<BilletBuyer>
|
||||
*/
|
||||
public function searchOrders(int $eventId, string $query): array
|
||||
{
|
||||
$index = self::INDEX_PREFIX.$eventId;
|
||||
|
||||
if ('' === $query) {
|
||||
return [];
|
||||
}
|
||||
|
||||
try {
|
||||
$results = $this->meilisearch->search($index, $query);
|
||||
$ids = array_map(fn (array $hit) => $hit['id'], $results['hits'] ?? []);
|
||||
|
||||
return $ids ? $this->em->getRepository(BilletBuyer::class)->findBy(['id' => $ids]) : [];
|
||||
} catch (\Throwable) {
|
||||
return [];
|
||||
}
|
||||
}
|
||||
|
||||
public function indexOrder(BilletBuyer $order): void
|
||||
{
|
||||
try {
|
||||
$index = self::INDEX_PREFIX.$order->getEvent()->getId();
|
||||
$this->meilisearch->createIndexIfNotExists($index);
|
||||
$this->meilisearch->addDocuments($index, [$this->toDocument($order)]);
|
||||
} catch (\Throwable) {
|
||||
// Meilisearch unavailable
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @return array<string, mixed>
|
||||
*/
|
||||
private function toDocument(BilletBuyer $order): array
|
||||
{
|
||||
$items = [];
|
||||
foreach ($order->getItems() as $item) {
|
||||
$items[] = $item->getBilletName().' x'.$item->getQuantity();
|
||||
}
|
||||
|
||||
return [
|
||||
'id' => $order->getId(),
|
||||
'orderNumber' => $order->getOrderNumber(),
|
||||
'reference' => $order->getReference(),
|
||||
'firstName' => $order->getFirstName(),
|
||||
'lastName' => $order->getLastName(),
|
||||
'email' => $order->getEmail(),
|
||||
'status' => $order->getStatus(),
|
||||
'totalHT' => $order->getTotalHTDecimal(),
|
||||
'items' => implode(', ', $items),
|
||||
'createdAt' => $order->getCreatedAt()->format('Y-m-d H:i'),
|
||||
];
|
||||
}
|
||||
}
|
||||
@@ -354,9 +354,92 @@
|
||||
</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 class="grid grid-cols-1 md:grid-cols-4 gap-4 mb-6">
|
||||
<div class="card-brutal p-4 text-center">
|
||||
<div class="text-[10px] font-black uppercase tracking-widest text-gray-500 mb-1">Commandes</div>
|
||||
<div class="text-2xl font-black">{{ event_total_orders }}</div>
|
||||
</div>
|
||||
<div class="card-brutal p-4 text-center">
|
||||
<div class="text-[10px] font-black uppercase tracking-widest text-gray-500 mb-1">Billets vendus</div>
|
||||
<div class="text-2xl font-black">{{ event_total_sold }}</div>
|
||||
</div>
|
||||
<div class="card-brutal p-4 text-center">
|
||||
<div class="text-[10px] font-black uppercase tracking-widest text-gray-500 mb-1">Chiffre d'affaires HT</div>
|
||||
<div class="text-2xl font-black text-indigo-600">{{ event_total_ht|number_format(2, ',', ' ') }} €</div>
|
||||
</div>
|
||||
<div class="card-brutal p-4 text-center bg-green-50">
|
||||
<div class="text-[10px] font-black uppercase tracking-widest text-gray-500 mb-1">Total percu</div>
|
||||
{% set total_commission_event = event_total_ht * (commission_rate / 100) %}
|
||||
<div class="text-2xl font-black text-green-600">{{ (event_total_ht - total_commission_event)|number_format(2, ',', ' ') }} €</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="card-brutal overflow-hidden">
|
||||
<div class="section-header flex justify-between items-center">
|
||||
<h2 class="text-[10px] font-black uppercase tracking-widest text-white">Commandes</h2>
|
||||
</div>
|
||||
<div class="p-6">
|
||||
<form method="get" action="{{ path('app_account_edit_event', {id: event.id, tab: 'stats'}) }}" class="flex gap-3 mb-6">
|
||||
<input type="hidden" name="tab" value="stats">
|
||||
<input type="text" name="q" value="{{ search_query }}" placeholder="Rechercher par nom, email, numero..." class="form-input focus:border-indigo-600 flex-1">
|
||||
<button type="submit" class="btn-brutal font-black uppercase text-xs tracking-widest hover:bg-indigo-600 hover:text-white transition-all">Rechercher</button>
|
||||
{% if search_query %}
|
||||
<a href="{{ path('app_account_edit_event', {id: event.id, tab: 'stats'}) }}" class="btn-brutal font-black uppercase text-xs tracking-widest hover:bg-gray-100 transition-all">Reset</a>
|
||||
{% endif %}
|
||||
</form>
|
||||
|
||||
{% if event_orders|length > 0 %}
|
||||
{% for order in event_orders %}
|
||||
<div class="border-2 border-gray-900 bg-white mb-3 p-4">
|
||||
<div class="flex flex-wrap items-center gap-3 mb-2">
|
||||
<span class="font-black uppercase text-sm">{{ order.orderNumber }}</span>
|
||||
{% if order.status == 'paid' %}
|
||||
<span class="badge-green text-[10px] font-black uppercase">Payee</span>
|
||||
{% elseif order.status == 'refunded' %}
|
||||
<span class="badge-yellow text-[10px] font-black uppercase">Remboursee</span>
|
||||
{% elseif order.status == 'cancelled' %}
|
||||
<span class="badge-red text-[10px] font-black uppercase">Annulee</span>
|
||||
{% else %}
|
||||
<span class="badge-yellow text-[10px] font-black uppercase">En attente</span>
|
||||
{% endif %}
|
||||
<span class="text-xs font-bold text-gray-400">{{ order.createdAt|date('d/m/Y H:i') }}</span>
|
||||
</div>
|
||||
|
||||
<div class="flex flex-wrap gap-4 text-sm font-bold text-gray-600 mb-2">
|
||||
<span>{{ order.firstName }} {{ order.lastName }}</span>
|
||||
<span class="text-gray-400">{{ order.email }}</span>
|
||||
</div>
|
||||
|
||||
<div class="flex flex-wrap items-center gap-3 mb-2">
|
||||
{% for item in order.items %}
|
||||
<span class="text-xs font-bold text-gray-500">{{ item.billetName }} x{{ item.quantity }}</span>
|
||||
{% endfor %}
|
||||
<span class="font-black text-sm text-indigo-600">{{ order.totalHTDecimal|number_format(2, ',', ' ') }} €</span>
|
||||
{% if order.paymentMethod %}
|
||||
<span class="text-xs text-gray-400 font-bold">{{ order.paymentMethod }}{% if order.cardBrand %} {{ order.cardBrand|upper }}{% endif %}{% if order.cardLast4 %} **** {{ order.cardLast4 }}{% endif %}</span>
|
||||
{% endif %}
|
||||
</div>
|
||||
|
||||
{% if order.status == 'paid' %}
|
||||
<div class="flex gap-2 mt-2">
|
||||
<form method="post" action="{{ path('app_account_event_cancel_order', {id: event.id, orderId: order.id}) }}" data-confirm="Annuler cette commande ? Les billets seront invalides.">
|
||||
<button type="submit" class="px-3 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">Annuler</button>
|
||||
</form>
|
||||
<form method="post" action="{{ path('app_account_event_refund_order', {id: event.id, orderId: order.id}) }}" data-confirm="Rembourser cette commande ? Le montant sera restitue au client.">
|
||||
<button type="submit" class="px-3 py-1 border-2 border-gray-900 bg-[#fabf04] text-xs font-black uppercase cursor-pointer hover:bg-yellow-500 transition-all">Rembourser</button>
|
||||
</form>
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endfor %}
|
||||
|
||||
<div class="mt-4">
|
||||
{{ knp_pagination_render(event_orders) }}
|
||||
</div>
|
||||
{% else %}
|
||||
<p class="text-gray-400 font-bold text-sm text-center py-8">{{ search_query ? 'Aucune commande trouvee.' : 'Aucune commande pour cet evenement.' }}</p>
|
||||
{% endif %}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
|
||||
Reference in New Issue
Block a user