feat: Implement promoter payout system for event revenue processing

- Add Payout model with associations to User and Event
- Create payout requests for completed events with proper earnings calculation
- Exclude refunded tickets from payout calculations
- Add promoter dashboard views for managing payouts
- Implement admin interface for processing payouts
- Integrate with Stripe for actual payment processing
- Add comprehensive tests for payout functionality

Co-authored-by: Qwen-Coder <qwen-coder@alibabacloud.com>
This commit is contained in:
kbe
2025-09-17 00:29:20 +02:00
parent 58141dca94
commit 59e1854803
20 changed files with 587 additions and 254 deletions

View File

@@ -151,6 +151,15 @@
<%= link_to promoter_event_path(event), class: "text-purple-600 hover:text-purple-800 text-xs font-medium" do %>
Gérer →
<% end %>
<% if event.event_ended? && event.can_request_payout? %>
<% if event.payout_status == "not_requested" %>
<%= link_to "Demander le paiement", promoter_payouts_path(event_id: event.id), method: :post,
class: "text-green-600 hover:text-green-800 text-xs font-medium" %>
<% else %>
<%= link_to "Voir le paiement", promoter_payouts_path,
class: "text-gray-600 hover:text-gray-800 text-xs font-medium" %>
<% end %>
<% end %>
</div>
</div>
<% end %>
@@ -165,41 +174,46 @@
</div>
</div>
<!-- Recent Orders -->
<% if @recent_orders.any? %>
<div class="bg-white rounded-2xl shadow-lg mb-8">
<div class="border-b border-gray-100 p-6">
<h2 class="text-xl font-bold text-gray-900">Commandes Récentes</h2>
<p class="text-gray-600 mt-1">Dernières commandes pour vos événements</p>
<!-- Recent orders for promoter events -->
<div class="mt-8">
<div class="flex justify-between items-center mb-4">
<h2 class="text-lg font-medium text-gray-900">Recent Orders</h2>
<%= link_to "View All Payouts", promoter_payouts_path, class: "text-sm font-medium text-indigo-600 hover:text-indigo-500" if current_user.promoter? %>
</div>
<div class="p-6">
<div class="overflow-x-auto">
<table class="w-full">
<thead>
<tr class="text-left border-b border-gray-200">
<th class="pb-3 text-sm font-medium text-gray-600">Événement</th>
<th class="pb-3 text-sm font-medium text-gray-600">Client</th>
<th class="pb-3 text-sm font-medium text-gray-600">Billets</th>
<th class="pb-3 text-sm font-medium text-gray-600">Montant</th>
<th class="pb-3 text-sm font-medium text-gray-600">Date</th>
</tr>
</thead>
<tbody class="divide-y divide-gray-100">
<% @recent_orders.each do |order| %>
<tr class="hover:bg-gray-50">
<td class="py-3 text-sm font-medium text-gray-900"><%= order.event.name %></td>
<td class="py-3 text-sm text-gray-700"><%= order.user.email %></td>
<td class="py-3 text-sm text-gray-700"><%= order.tickets.count %></td>
<td class="py-3 text-sm font-medium text-gray-900">€<%= order.total_amount_euros %></td>
<td class="py-3 text-sm text-gray-500"><%= order.created_at.strftime("%d/%m/%Y") %></td>
</tr>
<% end %>
</tbody>
</table>
<% if @recent_orders.any? %>
<div class="bg-white overflow-hidden shadow rounded-lg">
<ul class="divide-y divide-gray-200">
<% @recent_orders.each do |order| %>
<li>
<div class="px-4 py-4 flex items-center sm:px-6">
<div class="min-w-0 flex-1 sm:flex sm:items-center sm:justify-between">
<div class="truncate">
<div class="flex text-sm">
<p class="font-medium text-indigo-600 truncate"><%= order.event.name %></p>
<p class="ml-1 flex-shrink-0 font-normal text-gray-500">for <%= order.user.name.presence || order.user.email %></p>
</div>
<div class="mt-2 flex">
<div class="flex items-center text-sm text-gray-500">
<svg class="flex-shrink-0 mr-1.5 h-5 w-5 text-gray-400" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
<path fill-rule="evenodd" d="M6 2a1 1 0 00-1 1v1H4a2 2 0 00-2 2v10a2 2 0 002 2h12a2 2 0 002-2V6a2 2 0 00-2-2h-1V3a1 1 0 10-2 0v1H7V3a1 1 0 00-1-1zm0 5a1 1 0 000 2h8a1 1 0 100-2H6z" clip-rule="evenodd" />
</svg>
<span><%= order.created_at.strftime("%B %d, %Y") %></span>
</div>
</div>
</div>
<div class="ml-5 flex-shrink-0">
<p class="text-sm font-medium text-gray-900">€<%= order.total_amount_euros %></p>
</div>
</div>
</div>
</li>
<% end %>
</ul>
</div>
</div>
<% else %>
<p class="text-gray-500">No recent orders.</p>
<% end %>
</div>
<% end %>
<% end %>
<!-- Draft orders needing payment -->