feat: replace Stripe Global Payouts with manual bank transfer system for France compliance

- Replace Stripe automatic payouts with manual admin-processed bank transfers
- Add banking information fields (IBAN, bank name, account holder) to User model
- Implement manual payout workflow: pending → approved → processing → completed
- Add comprehensive admin interface for payout review and processing
- Update Payout model with manual processing fields and workflow methods
- Add transfer reference tracking and rejection/failure handling
- Consolidate all migration fragments into clean "create" migrations
- Add comprehensive documentation for manual payout workflow
- Fix Event payout_status enum definition and database column issues

This addresses France's lack of Stripe Global Payouts support by implementing
a complete manual bank transfer workflow while maintaining audit trails and
proper admin controls.

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
kbe
2025-09-17 11:55:07 +02:00
parent 3c1e17c2af
commit 1889ee7fb2
20 changed files with 838 additions and 141 deletions

View File

@@ -1,32 +1,77 @@
class Admin::PayoutsController < ApplicationController
before_action :authenticate_user!
before_action :ensure_admin!
before_action :set_payout, only: [:show, :approve, :reject, :mark_processing, :mark_completed, :mark_failed]
def index
@payouts = Payout.pending.includes(:user, :event).order(created_at: :asc).page(params[:page])
@pending_payouts = Payout.pending.includes(:user, :event).order(created_at: :asc)
@approved_payouts = Payout.approved.includes(:user, :event).order(created_at: :asc)
@processing_payouts = Payout.processing.includes(:user, :event).order(created_at: :asc)
@completed_payouts = Payout.completed.includes(:user, :event).order(created_at: :desc).limit(10)
end
def show
@payout = Payout.find(params[:id])
@service = PayoutService.new(@payout)
@transfer_summary = @service.generate_transfer_summary
@banking_errors = @service.validate_banking_info
end
def process
@payout = Payout.find(params[:id])
if @payout.pending? && @payout.can_process?
begin
PayoutService.new(@payout).process!
redirect_to admin_payouts_path, notice: "Payout processed successfully."
rescue => e
redirect_to admin_payouts_path, alert: "Failed to process payout: #{e.message}"
end
def approve
if @payout.approve!(current_user)
redirect_to admin_payout_path(@payout), notice: "Payout approved successfully."
else
redirect_to admin_payouts_path, alert: "Cannot process this payout."
redirect_to admin_payout_path(@payout), alert: "Cannot approve this payout."
end
end
def reject
reason = params[:rejection_reason].presence || "No reason provided"
if @payout.reject!(current_user, reason)
redirect_to admin_payouts_path, notice: "Payout rejected."
else
redirect_to admin_payout_path(@payout), alert: "Cannot reject this payout."
end
end
def mark_processing
transfer_reference = params[:bank_transfer_reference]
if @payout.mark_processing!(current_user, transfer_reference)
redirect_to admin_payout_path(@payout), notice: "Payout marked as processing."
else
redirect_to admin_payout_path(@payout), alert: "Cannot mark payout as processing."
end
end
def mark_completed
transfer_reference = params[:bank_transfer_reference]
if @payout.mark_completed!(current_user, transfer_reference)
redirect_to admin_payouts_path, notice: "Payout completed successfully."
else
redirect_to admin_payout_path(@payout), alert: "Cannot mark payout as completed."
end
end
def mark_failed
reason = params[:failure_reason].presence || "Transfer failed"
if @payout.mark_failed!(current_user, reason)
redirect_to admin_payouts_path, notice: "Payout marked as failed."
else
redirect_to admin_payout_path(@payout), alert: "Cannot mark payout as failed."
end
end
# Legacy method - redirect to new workflow
def process
@payout = Payout.find(params[:id])
redirect_to admin_payout_path(@payout), alert: "Use the new manual payout workflow."
end
private
def set_payout
@payout = Payout.find(params[:id])
end
def ensure_admin!
# For now, we'll just check if the user has a stripe account
# In a real app, you'd have an admin role check

View File

@@ -17,12 +17,12 @@ class Event < ApplicationRecord
}, default: :draft
enum :payout_status, {
not_requested: 0,
pending_request: 0,
requested: 1,
processing: 2,
completed: 3,
failed: 4
}, default: :not_requested
}, default: :pending_request
# === Relations ===
belongs_to :user

View File

@@ -2,13 +2,16 @@ class Payout < ApplicationRecord
# === Relations ===
belongs_to :user
belongs_to :event
belongs_to :processed_by, class_name: 'User', optional: true
# === Enums ===
enum :status, {
pending: 0, # Payout requested but not processed
processing: 1, # Payout being processed
completed: 2, # Payout successfully completed
failed: 3 # Payout failed
pending: 0, # Payout requested but not reviewed
approved: 1, # Payout approved by admin, ready for transfer
processing: 2, # Payout being processed (bank transfer initiated)
completed: 3, # Payout successfully completed
failed: 4, # Payout failed
rejected: 5 # Payout rejected by admin
}, default: :pending
# === Validations ===
@@ -45,7 +48,10 @@ class Payout < ApplicationRecord
# === Scopes ===
scope :completed, -> { where(status: :completed) }
scope :pending, -> { where(status: :pending) }
scope :approved, -> { where(status: :approved) }
scope :processing, -> { where(status: :processing) }
scope :rejected, -> { where(status: :rejected) }
scope :failed, -> { where(status: :failed) }
# === Callbacks ===
after_create :calculate_refunded_orders_count
@@ -72,15 +78,74 @@ class Payout < ApplicationRecord
net_amount_cents / 100.0
end
# Check if payout can be processed
def can_process?
pending? && amount_cents > 0
# Check if payout can be approved (was pending)
def can_approve?
pending? && amount_cents > 0 && user.has_complete_banking_info?
end
# Process the payout through Stripe
def process_payout!
service = PayoutService.new(self)
service.process!
# Check if payout can be manually processed (was approved)
def can_process?
approved? && amount_cents > 0
end
# Check if payout can be rejected
def can_reject?
pending?
end
# Approve the payout for manual processing
def approve!(admin_user)
return false unless can_approve?
update!(
status: :approved,
processed_by: admin_user,
processed_at: Time.current
)
end
# Reject the payout with reason
def reject!(admin_user, reason)
return false unless can_reject?
update!(
status: :rejected,
processed_by: admin_user,
processed_at: Time.current,
rejection_reason: reason
)
end
# Mark as processing (bank transfer initiated)
def mark_processing!(admin_user, transfer_reference = nil)
return false unless can_process?
update!(
status: :processing,
processed_by: admin_user,
processed_at: Time.current,
bank_transfer_reference: transfer_reference
)
end
# Mark as completed (bank transfer confirmed)
def mark_completed!(admin_user, transfer_reference = nil)
return false unless processing?
update!(
status: :completed,
processed_by: admin_user,
processed_at: Time.current,
bank_transfer_reference: transfer_reference || bank_transfer_reference
)
update_earnings_status
end
# Mark as failed
def mark_failed!(admin_user, reason)
return false unless processing?
update!(
status: :failed,
processed_by: admin_user,
processed_at: Time.current,
rejection_reason: reason
)
end
public

View File

@@ -31,6 +31,11 @@ class User < ApplicationRecord
validates :first_name, length: { minimum: 2, maximum: 50, allow_blank: true }
validates :company_name, length: { minimum: 2, maximum: 100, allow_blank: true }
# Banking information validations
validates :iban, format: { with: /\A[A-Z]{2}[0-9]{2}[A-Z0-9]{4}[0-9]{7}([A-Z0-9]?){0,16}\z/, message: "must be a valid IBAN format" }, allow_blank: true
validates :bank_name, length: { minimum: 2, maximum: 100 }, allow_blank: true
validates :account_holder_name, length: { minimum: 2, maximum: 100 }, allow_blank: true
# Onboarding methods
def needs_onboarding?
!onboarding_completed?
@@ -65,7 +70,17 @@ class User < ApplicationRecord
end
def can_receive_payouts?
stripe_connected_account_id.present? && stripe_connect_verified?
has_complete_banking_info?
end
# Banking information methods
def has_complete_banking_info?
iban.present? && bank_name.present? && account_holder_name.present?
end
def banking_info_summary
return "No banking information" unless has_complete_banking_info?
"#{account_holder_name} - #{bank_name} - #{iban}"
end
private

View File

@@ -3,36 +3,51 @@ class PayoutService
@payout = payout
end
# Legacy method for backward compatibility - now redirects to manual workflow
def process!
return unless @payout.can_process?
Rails.logger.warn "PayoutService#process! called - manual processing required for payout #{@payout.id}"
raise "Automatic payout processing is disabled. Use manual workflow in admin interface."
end
@payout.update!(status: :processing)
# Generate payout summary for manual transfer
def generate_transfer_summary
return nil unless @payout.approved? || @payout.processing?
begin
net_amount = @payout.amount_cents - @payout.fee_cents
transfer = Stripe::Transfer.create({
amount: (net_amount / 100.0).to_i,
currency: "eur",
destination: @payout.user.stripe_connected_account_id,
description: "Payout for event #{@payout.event.name}",
metadata: { payout_id: @payout.id, event_id: @payout.event_id }
}, idempotency_key: SecureRandom.uuid)
{
payout_id: @payout.id,
recipient: @payout.user.name,
account_holder: @payout.user.account_holder_name,
bank_name: @payout.user.bank_name,
iban: @payout.user.iban,
amount_euros: @payout.net_amount_euros,
description: "Payout for event: #{@payout.event.name}",
event_name: @payout.event.name,
event_date: @payout.event.date,
total_orders: @payout.total_orders_count,
refunded_orders: @payout.refunded_orders_count
}
end
@payout.update!(
status: :completed,
stripe_payout_id: transfer.id
)
# Validate banking information before processing
def validate_banking_info
errors = []
user = @payout.user
update_earnings_status
rescue Stripe::StripeError => e
@payout.update!(status: :failed)
Rails.logger.error "Stripe payout failed for payout #{@payout.id}: #{e.message}"
raise e
end
errors << "Missing IBAN" unless user.iban.present?
errors << "Missing bank name" unless user.bank_name.present?
errors << "Missing account holder name" unless user.account_holder_name.present?
errors << "Invalid IBAN format" if user.iban.present? && !valid_iban?(user.iban)
errors
end
private
def valid_iban?(iban)
# Basic IBAN validation (simplified)
iban.match?(/\A[A-Z]{2}[0-9]{2}[A-Z0-9]{4}[0-9]{7}([A-Z0-9]?){0,16}\z/)
end
def update_earnings_status
@payout.event.earnings.where(status: 0).update_all(status: 1) # pending to paid
end

View File

@@ -0,0 +1,96 @@
<table class="min-w-full divide-y divide-gray-200">
<thead class="bg-gray-50">
<tr>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Event</th>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Promoter</th>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Banking Info</th>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Amount</th>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Status</th>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Date</th>
<% if show_actions %>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Actions</th>
<% end %>
</tr>
</thead>
<tbody class="bg-white divide-y divide-gray-200">
<% payouts.each do |payout| %>
<tr>
<td class="px-6 py-4 whitespace-nowrap">
<div class="text-sm font-medium text-gray-900"><%= payout.event.name %></div>
<div class="text-sm text-gray-500"><%= payout.event.date.strftime("%b %d, %Y") if payout.event.date %></div>
</td>
<td class="px-6 py-4 whitespace-nowrap">
<div class="text-sm text-gray-900"><%= payout.user.name.presence || payout.user.email %></div>
</td>
<td class="px-6 py-4 whitespace-nowrap">
<% if payout.user.has_complete_banking_info? %>
<div class="text-sm text-gray-900">✅ Complete</div>
<div class="text-sm text-gray-500"><%= payout.user.bank_name %></div>
<% else %>
<div class="text-sm text-red-600">❌ Incomplete</div>
<div class="text-sm text-gray-500">Missing banking info</div>
<% end %>
</td>
<td class="px-6 py-4 whitespace-nowrap">
<div class="text-sm text-gray-900">€<%= payout.amount_euros %></div>
<div class="text-sm text-gray-500">Net: €<%= payout.net_amount_euros %></div>
</td>
<td class="px-6 py-4 whitespace-nowrap">
<% case payout.status %>
<% when 'pending' %>
<span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-yellow-100 text-yellow-800">
Pending Review
</span>
<% when 'approved' %>
<span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-blue-100 text-blue-800">
Approved
</span>
<% when 'processing' %>
<span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-indigo-100 text-indigo-800">
Processing
</span>
<% when 'completed' %>
<span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-100 text-green-800">
Completed
</span>
<% when 'failed' %>
<span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-red-100 text-red-800">
Failed
</span>
<% when 'rejected' %>
<span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-gray-100 text-gray-800">
Rejected
</span>
<% end %>
</td>
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500">
<%= payout.created_at.strftime("%b %d, %Y") %>
<% if payout.processed_at %>
<div class="text-xs text-gray-400">Processed: <%= payout.processed_at.strftime("%b %d") %></div>
<% end %>
</td>
<% if show_actions %>
<td class="px-6 py-4 whitespace-nowrap text-sm font-medium">
<%= link_to "View", admin_payout_path(payout), class: "text-indigo-600 hover:text-indigo-900" %>
<% case section %>
<% when 'pending' %>
<% if payout.can_approve? %>
<%= link_to "Approve", approve_admin_payout_path(payout), method: :post,
class: "text-green-600 hover:text-green-900 ml-2",
data: { confirm: "Approve this payout for transfer?" } %>
<% end %>
<% when 'approved' %>
<%= link_to "Start Transfer", mark_processing_admin_payout_path(payout), method: :post,
class: "text-blue-600 hover:text-blue-900 ml-2",
data: { confirm: "Mark as processing (transfer initiated)?" } %>
<% when 'processing' %>
<%= link_to "Complete", mark_completed_admin_payout_path(payout), method: :post,
class: "text-green-600 hover:text-green-900 ml-2",
data: { confirm: "Mark transfer as completed?" } %>
<% end %>
</td>
<% end %>
</tr>
<% end %>
</tbody>
</table>

View File

@@ -1,76 +1,49 @@
<div class="container mx-auto px-4 py-8">
<div class="flex justify-between items-center mb-6">
<h1 class="text-3xl font-bold text-gray-900">Admin Payouts</h1>
<h1 class="text-3xl font-bold text-gray-900">Manual Payout Administration</h1>
</div>
<% if @payouts.any? %>
<div class="bg-white rounded-lg shadow overflow-hidden">
<table class="min-w-full divide-y divide-gray-200">
<thead class="bg-gray-50">
<tr>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Event</th>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Promoter</th>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Amount</th>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Status</th>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Date</th>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Actions</th>
</tr>
</thead>
<tbody class="bg-white divide-y divide-gray-200">
<% @payouts.each do |payout| %>
<tr>
<td class="px-6 py-4 whitespace-nowrap">
<div class="text-sm font-medium text-gray-900"><%= payout.event.name %></div>
</td>
<td class="px-6 py-4 whitespace-nowrap">
<div class="text-sm text-gray-900"><%= payout.user.name.presence || payout.user.email %></div>
</td>
<td class="px-6 py-4 whitespace-nowrap">
<div class="text-sm text-gray-900">€<%= payout.amount_euros %></div>
<div class="text-sm text-gray-500">Net: €<%= payout.net_amount_euros %> (Fee: €<%= payout.fee_euros %>)</div>
</td>
<td class="px-6 py-4 whitespace-nowrap">
<% case payout.status %>
<% when 'pending' %>
<span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-yellow-100 text-yellow-800">
Pending
</span>
<% when 'processing' %>
<span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-blue-100 text-blue-800">
Processing
</span>
<% when 'completed' %>
<span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-100 text-green-800">
Completed
</span>
<% when 'failed' %>
<span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-red-100 text-red-800">
Failed
</span>
<% end %>
</td>
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500">
<%= payout.created_at.strftime("%b %d, %Y") %>
</td>
<td class="px-6 py-4 whitespace-nowrap text-sm font-medium">
<% if payout.can_process? %>
<%= button_to "Process", admin_payout_path(payout), method: :post,
class: "text-indigo-600 hover:text-indigo-900 bg-indigo-100 hover:bg-indigo-200 px-3 py-1 rounded" %>
<% end %>
<%= link_to "View", promoter_payout_path(payout), class: "text-indigo-600 hover:text-indigo-900 ml-2" %>
</td>
</tr>
<% end %>
</tbody>
</table>
</div>
<% if @payouts.respond_to?(:total_pages) %>
<div class="mt-6">
<%= paginate @payouts %>
<!-- Pending Payouts - Require Review -->
<% if @pending_payouts.any? %>
<div class="mb-8">
<h2 class="text-xl font-semibold text-gray-900 mb-4">📋 Pending Review (<%= @pending_payouts.count %>)</h2>
<div class="bg-white rounded-lg shadow overflow-hidden">
<%= render partial: 'payout_table', locals: { payouts: @pending_payouts, show_actions: true, section: 'pending' } %>
</div>
<% end %>
<% else %>
</div>
<% end %>
<!-- Approved Payouts - Ready for Transfer -->
<% if @approved_payouts.any? %>
<div class="mb-8">
<h2 class="text-xl font-semibold text-gray-900 mb-4">✅ Approved - Ready for Transfer (<%= @approved_payouts.count %>)</h2>
<div class="bg-white rounded-lg shadow overflow-hidden">
<%= render partial: 'payout_table', locals: { payouts: @approved_payouts, show_actions: true, section: 'approved' } %>
</div>
</div>
<% end %>
<!-- Processing Payouts - Transfer Initiated -->
<% if @processing_payouts.any? %>
<div class="mb-8">
<h2 class="text-xl font-semibold text-gray-900 mb-4">🔄 Processing - Transfer in Progress (<%= @processing_payouts.count %>)</h2>
<div class="bg-white rounded-lg shadow overflow-hidden">
<%= render partial: 'payout_table', locals: { payouts: @processing_payouts, show_actions: true, section: 'processing' } %>
</div>
</div>
<% end %>
<!-- Recent Completed Payouts -->
<% if @completed_payouts.any? %>
<div class="mb-8">
<h2 class="text-xl font-semibold text-gray-900 mb-4">✨ Recently Completed</h2>
<div class="bg-white rounded-lg shadow overflow-hidden">
<%= render partial: 'payout_table', locals: { payouts: @completed_payouts, show_actions: false, section: 'completed' } %>
</div>
</div>
<% end %>
<% if @pending_payouts.empty? && @approved_payouts.empty? && @processing_payouts.empty? && @completed_payouts.empty? %>
<div class="bg-white rounded-lg shadow p-6 text-center">
<p class="text-gray-500">No payouts found.</p>
</div>

View File

@@ -1,2 +1,208 @@
<h1>Admin::Payouts#show</h1>
<p>Find me in app/views/admin/payouts/show.html.erb</p>
<div class="container mx-auto px-4 py-8">
<div class="flex justify-between items-center mb-6">
<h1 class="text-3xl font-bold text-gray-900">Payout Details #<%= @payout.id %></h1>
<%= link_to "← Back to Payouts", admin_payouts_path, class: "text-indigo-600 hover:text-indigo-900" %>
</div>
<div class="grid grid-cols-1 lg:grid-cols-2 gap-8">
<!-- Payout Information -->
<div class="bg-white rounded-lg shadow p-6">
<h2 class="text-xl font-semibold text-gray-900 mb-4">Payout Information</h2>
<div class="space-y-4">
<div>
<label class="block text-sm font-medium text-gray-500">Status</label>
<% case @payout.status %>
<% when 'pending' %>
<span class="px-3 py-1 text-sm font-semibold rounded-full bg-yellow-100 text-yellow-800">
Pending Review
</span>
<% when 'approved' %>
<span class="px-3 py-1 text-sm font-semibold rounded-full bg-blue-100 text-blue-800">
Approved - Ready for Transfer
</span>
<% when 'processing' %>
<span class="px-3 py-1 text-sm font-semibold rounded-full bg-indigo-100 text-indigo-800">
Processing
</span>
<% when 'completed' %>
<span class="px-3 py-1 text-sm font-semibold rounded-full bg-green-100 text-green-800">
Completed
</span>
<% when 'failed' %>
<span class="px-3 py-1 text-sm font-semibold rounded-full bg-red-100 text-red-800">
Failed
</span>
<% when 'rejected' %>
<span class="px-3 py-1 text-sm font-semibold rounded-full bg-gray-100 text-gray-800">
Rejected
</span>
<% end %>
</div>
<div>
<label class="block text-sm font-medium text-gray-500">Event</label>
<p class="text-gray-900"><%= @payout.event.name %></p>
<p class="text-sm text-gray-500"><%= @payout.event.date.strftime("%B %d, %Y") if @payout.event.date %></p>
</div>
<div>
<label class="block text-sm font-medium text-gray-500">Promoter</label>
<p class="text-gray-900"><%= @payout.user.name.presence || @payout.user.email %></p>
<p class="text-sm text-gray-500"><%= @payout.user.email %></p>
</div>
<div class="grid grid-cols-2 gap-4">
<div>
<label class="block text-sm font-medium text-gray-500">Gross Amount</label>
<p class="text-lg font-semibold text-gray-900">€<%= @payout.amount_euros %></p>
</div>
<div>
<label class="block text-sm font-medium text-gray-500">Platform Fee</label>
<p class="text-lg font-semibold text-gray-900">€<%= @payout.fee_euros %></p>
</div>
</div>
<div>
<label class="block text-sm font-medium text-gray-500">Net Amount (To Transfer)</label>
<p class="text-2xl font-bold text-green-600">€<%= @payout.net_amount_euros %></p>
</div>
<div class="grid grid-cols-2 gap-4">
<div>
<label class="block text-sm font-medium text-gray-500">Total Orders</label>
<p class="text-gray-900"><%= @payout.total_orders_count %></p>
</div>
<div>
<label class="block text-sm font-medium text-gray-500">Refunded Orders</label>
<p class="text-gray-900"><%= @payout.refunded_orders_count %></p>
</div>
</div>
<div>
<label class="block text-sm font-medium text-gray-500">Requested</label>
<p class="text-gray-900"><%= @payout.created_at.strftime("%B %d, %Y at %I:%M %p") %></p>
</div>
<% if @payout.processed_at %>
<div>
<label class="block text-sm font-medium text-gray-500">Processed</label>
<p class="text-gray-900"><%= @payout.processed_at.strftime("%B %d, %Y at %I:%M %p") %></p>
<% if @payout.processed_by %>
<p class="text-sm text-gray-500">by <%= @payout.processed_by.name.presence || @payout.processed_by.email %></p>
<% end %>
</div>
<% end %>
<% if @payout.bank_transfer_reference.present? %>
<div>
<label class="block text-sm font-medium text-gray-500">Transfer Reference</label>
<p class="text-gray-900 font-mono"><%= @payout.bank_transfer_reference %></p>
</div>
<% end %>
<% if @payout.rejection_reason.present? %>
<div>
<label class="block text-sm font-medium text-gray-500">Rejection/Failure Reason</label>
<p class="text-red-600"><%= @payout.rejection_reason %></p>
</div>
<% end %>
</div>
</div>
<!-- Banking Information -->
<div class="bg-white rounded-lg shadow p-6">
<h2 class="text-xl font-semibold text-gray-900 mb-4">Banking Information</h2>
<% if @banking_errors.any? %>
<div class="mb-4 p-4 bg-red-50 border border-red-200 rounded-md">
<h3 class="text-sm font-medium text-red-800">Banking Information Issues:</h3>
<ul class="mt-2 text-sm text-red-700">
<% @banking_errors.each do |error| %>
<li>• <%= error %></li>
<% end %>
</ul>
</div>
<% end %>
<% if @transfer_summary %>
<div class="space-y-4">
<div>
<label class="block text-sm font-medium text-gray-500">Account Holder</label>
<p class="text-gray-900"><%= @transfer_summary[:account_holder] %></p>
</div>
<div>
<label class="block text-sm font-medium text-gray-500">Bank Name</label>
<p class="text-gray-900"><%= @transfer_summary[:bank_name] %></p>
</div>
<div>
<label class="block text-sm font-medium text-gray-500">IBAN</label>
<p class="text-gray-900 font-mono"><%= @transfer_summary[:iban] %></p>
</div>
<div class="p-4 bg-blue-50 border border-blue-200 rounded-md">
<h3 class="text-sm font-medium text-blue-800">Transfer Instructions</h3>
<div class="mt-2 text-sm text-blue-700">
<p><strong>Amount:</strong> €<%= @transfer_summary[:amount_euros] %></p>
<p><strong>Reference:</strong> Payout #<%= @transfer_summary[:payout_id] %> - <%= @transfer_summary[:event_name] %></p>
</div>
</div>
</div>
<% else %>
<div class="text-center text-gray-500 py-8">
<p>Banking information not available for display.</p>
</div>
<% end %>
</div>
</div>
<!-- Actions -->
<div class="mt-8 bg-white rounded-lg shadow p-6">
<h2 class="text-xl font-semibold text-gray-900 mb-4">Actions</h2>
<div class="flex flex-wrap gap-4">
<% if @payout.can_approve? %>
<%= button_to "✅ Approve Payout", approve_admin_payout_path(@payout), method: :post,
class: "bg-green-600 hover:bg-green-700 text-white px-4 py-2 rounded-md font-medium",
data: { confirm: "Approve this payout for manual bank transfer?" } %>
<% end %>
<% if @payout.can_reject? %>
<%= form_with url: reject_admin_payout_path(@payout), method: :post, local: true, class: "flex gap-2" do |form| %>
<%= form.text_field :rejection_reason, placeholder: "Rejection reason...", required: true,
class: "border border-gray-300 rounded-md px-3 py-2" %>
<%= form.submit "❌ Reject", class: "bg-red-600 hover:bg-red-700 text-white px-4 py-2 rounded-md font-medium",
data: { confirm: "Reject this payout?" } %>
<% end %>
<% end %>
<% if @payout.can_process? %>
<%= form_with url: mark_processing_admin_payout_path(@payout), method: :post, local: true, class: "flex gap-2" do |form| %>
<%= form.text_field :bank_transfer_reference, placeholder: "Transfer reference (optional)",
class: "border border-gray-300 rounded-md px-3 py-2" %>
<%= form.submit "🔄 Mark as Processing", class: "bg-blue-600 hover:bg-blue-700 text-white px-4 py-2 rounded-md font-medium",
data: { confirm: "Mark as processing (bank transfer initiated)?" } %>
<% end %>
<% end %>
<% if @payout.processing? %>
<%= form_with url: mark_completed_admin_payout_path(@payout), method: :post, local: true, class: "flex gap-2" do |form| %>
<%= form.text_field :bank_transfer_reference, placeholder: "Final transfer reference",
value: @payout.bank_transfer_reference,
class: "border border-gray-300 rounded-md px-3 py-2" %>
<%= form.submit "✅ Mark as Completed", class: "bg-green-600 hover:bg-green-700 text-white px-4 py-2 rounded-md font-medium",
data: { confirm: "Confirm transfer completion?" } %>
<% end %>
<%= form_with url: mark_failed_admin_payout_path(@payout), method: :post, local: true, class: "flex gap-2" do |form| %>
<%= form.text_field :failure_reason, placeholder: "Failure reason...", required: true,
class: "border border-gray-300 rounded-md px-3 py-2" %>
<%= form.submit "❌ Mark as Failed", class: "bg-red-600 hover:bg-red-700 text-white px-4 py-2 rounded-md font-medium",
data: { confirm: "Mark transfer as failed?" } %>
<% end %>
<% end %>
</div>
</div>
</div>