4 Commits

Author SHA1 Message Date
kbe
fa99a167a5 style: link content 2025-09-06 00:37:25 +02:00
kbe
9b33b73bb4 style: Clean up whitespace in tickets controller
- Remove extra blank lines and trailing spaces
- Improve code formatting consistency
- No functional changes

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

Co-Authored-By: Claude <noreply@anthropic.com>
2025-09-06 00:36:25 +02:00
kbe
bc47027c22 refactor: Convert ticket views to use only Tailwind CSS
- Rewrite ticket show view to use pure Tailwind CSS classes
- Update color scheme from gray-* to slate-* for modern look
- Replace indigo gradients with violet for better consistency
- Enhance spacing, typography, and visual hierarchy
- Add ticket_view route and controller action for PDF-like display
- Implement responsive QR code display with proper sizing
- Update status badge colors for better semantic meaning
- Improve accessibility with better button layouts and focus states

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

Co-Authored-By: Claude <noreply@anthropic.com>
2025-09-06 00:33:24 +02:00
kbe
7ef934d8a8 fix: Replace Prawn with Grover for PDF ticket generation
- Replace Prawn PDF generation with Grover (Chrome headless) for better compatibility
- Add HTML-based ticket template with embedded CSS styling
- Implement robust Grover loading with fallback to HTML download
- Add QR code generation methods to Ticket model
- Remove legacy TicketPdfGenerator service and tests
- Update PDF generation in TicketsController with proper error handling

The new implementation provides:
- Better HTML/CSS rendering for ticket layouts
- More reliable PDF generation using Chrome engine
- Fallback mechanism for better user experience
- Cleaner separation of template rendering and PDF conversion

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

Co-Authored-By: Claude <noreply@anthropic.com>
2025-09-06 00:04:02 +02:00
75 changed files with 2081 additions and 5533 deletions

View File

@@ -1,18 +1,18 @@
# Application data
RAILS_ENV=production
RAILS_ENV=development
SECRET_KEY_BASE=a3f5c6e7b8d9e0f1a2b3c4d5e6f7g8h9i0j1k2l3m4n5o6p7q8r9s0t1u2v3w4x5y6z7
DEVISE_SECRET_KEY=your_devise_secret_key_here
APP_NAME=Aperonight
# Database Configuration for production and development
# DB_HOST=127.0.0.1
# DB_PORT=3306
DB_HOST=localhost
DB_ROOT_PASSWORD=root
DB_DATABASE=aperonight
DB_USERNAME=root
DB_PASSWORD=root
# Test database
DB_TEST_ADAPTER=sqlite3
DB_TEST_DATABASE=aperonight_test
DB_TEST_USERNAME=root
DB_TEST_USERNAME=root
@@ -28,6 +28,14 @@ SMTP_PORT=1025
# SMTP_DOMAIN=localhost
SMTP_AUTHENTICATION=plain
SMTP_ENABLE_STARTTLS=false
# Production SMTP Configuration (set these in .env.production)
# SMTP_ADDRESS=smtp.example.com
# SMTP_PORT=587
# SMTP_USERNAME=your_smtp_username
# SMTP_PASSWORD=your_smtp_password
# SMTP_AUTHENTICATION=plain
# SMTP_DOMAIN=example.com
# SMTP_STARTTLS=true
# Application variables

View File

@@ -2,50 +2,43 @@
## 📋 Todo
### High Priority
- [ ] feat: Check-in system with QR code scanning
### Medium Priority
- [ ] feat: Promoter system with event creation, ticket types creation and metrics display
- [ ] feat: Multiple ticket types (early bird, VIP, general admission)
- [ ] feat: Refund management system
- [ ] feat: Real-time sales analytics dashboard
- [ ] feat: Guest checkout without account creation
- [ ] feat: Seat selection with interactive venue maps
- [ ] feat: Dynamic pricing based on demand
### Low Priority
- [ ] feat: SMS integration for ticket delivery and updates
- [ ] feat: Mobile wallet integration
- [ ] feat: Multi-currency support
- [ ] feat: Event updates communication system
- [ ] feat: Bulk operations for group bookings
- [ ] feat: Fraud prevention and bot protection
- [ ] feat: Social login options
- [ ] feat: Event recommendations system
### Design & Infrastructure
- [ ] style: Rewrite design system
- [ ] refactor: Rewrite design mockup
- [ ] Set up project infrastructure
- [ ] Design user interface mockups
- [ ] Create user dashboard
- [ ] Implement data persistence
- [ ] Add responsive design
- [ ] Write unit tests
- [ ] Set up CI/CD pipeline
- [ ] Add error handling
- [ ] Implement search functionality
- [ ] Add user profile management
- [ ] Create admin panel
- [ ] Optimize performance
- [ ] Add documentation
- [ ] Security audit
- [ ] Deploy to production
## 🚧 Doing
- [ ] feat: Page to display all tickets for an event
- [ ] feat: Add a link into notification email to order page that display all tickets
- [ ] refactor: Moving checkout to OrdersController
## ✅ Done
- [x] Initialize git repository
- [x] Set up development environment
- [x] Create project structure
- [x] Install dependencies
- [x] Configure build tools
- [x] Set up linting rules
- [x] Create initial README
- [x] Set up version control
- [x] Configure development server
- [x] Establish coding standards
- [x] Set up package.json
- [x] Create .gitignore file
- [x] Initialize npm project
- [x] Set up basic folder structure
- [x] Configure environment variables
- [x] Create authentication system
- [x] Implement user registration
- [x] Add login functionality
- [x] refactor: Moving checkout to OrdersController
- [x] feat: Payment gateway integration (Stripe) - PayPal not implemented
- [x] feat: Digital tickets with QR codes
- [x] feat: Ticket inventory management and capacity limits
- [x] feat: Event discovery with search and filtering
- [x] feat: Email notifications (purchase confirmations, event reminders)

View File

@@ -87,8 +87,7 @@ gem "kaminari-tailwind", "~> 0.1.0"
gem "stripe", "~> 15.5"
# PDF generation for tickets
gem "prawn", "~> 2.5"
gem "prawn-qrcode", "~> 0.5"
gem "grover"
# QR code generation
gem "rqrcode", "~> 3.1"

View File

@@ -127,6 +127,8 @@ GEM
raabro (~> 1.4)
globalid (1.2.1)
activesupport (>= 6.1)
grover (1.2.3)
nokogiri (~> 1)
i18n (1.14.7)
concurrent-ruby (~> 1.0)
io-console (0.8.1)
@@ -221,16 +223,8 @@ GEM
parser (3.3.9.0)
ast (~> 2.4.1)
racc
pdf-core (0.10.0)
pp (0.6.2)
prettyprint
prawn (2.5.0)
matrix (~> 0.4)
pdf-core (~> 0.10.0)
ttfunk (~> 1.8)
prawn-qrcode (0.5.2)
prawn (>= 1)
rqrcode (>= 1.0.0)
prettyprint (0.2.0)
prism (1.4.0)
propshaft (1.2.1)
@@ -378,8 +372,6 @@ GEM
thruster (0.1.15-aarch64-linux)
thruster (0.1.15-x86_64-linux)
timeout (0.4.3)
ttfunk (1.8.0)
bigdecimal (~> 3.1)
turbo-rails (2.0.16)
actionpack (>= 7.1.0)
railties (>= 7.1.0)
@@ -423,6 +415,7 @@ DEPENDENCIES
debug
devise (~> 4.9)
dotenv-rails
grover
jbuilder
jsbundling-rails
kamal
@@ -431,8 +424,6 @@ DEPENDENCIES
minitest-reporters (~> 1.7)
mocha
mysql2 (~> 0.5)
prawn (~> 2.5)
prawn-qrcode (~> 0.5)
propshaft
puma (>= 5.0)
rails (~> 8.0.2, >= 8.0.2.1)

View File

@@ -1,185 +0,0 @@
// Self-contained QR Code Generator
// No external dependencies required
class QRCodeGenerator {
constructor() {
// QR Code error correction levels
this.errorCorrectionLevels = {
L: 1, // Low ~7%
M: 0, // Medium ~15%
Q: 3, // Quartile ~25%
H: 2 // High ~30%
};
// Mode indicators
this.modes = {
NUMERIC: 1,
ALPHANUMERIC: 2,
BYTE: 4,
KANJI: 8
};
}
// Generate QR code as SVG
generateSVG(text, options = {}) {
const size = options.size || 200;
const margin = options.margin || 4;
const errorCorrection = options.errorCorrection || 'M';
try {
const qrData = this.createQRData(text, errorCorrection);
const moduleSize = (size - 2 * margin) / qrData.length;
let svg = `<svg width="${size}" height="${size}" viewBox="0 0 ${size} ${size}" xmlns="http://www.w3.org/2000/svg">`;
svg += `<rect width="${size}" height="${size}" fill="white"/>`;
for (let row = 0; row < qrData.length; row++) {
for (let col = 0; col < qrData[row].length; col++) {
if (qrData[row][col]) {
const x = margin + col * moduleSize;
const y = margin + row * moduleSize;
svg += `<rect x="${x}" y="${y}" width="${moduleSize}" height="${moduleSize}" fill="black"/>`;
}
}
}
svg += '</svg>';
return svg;
} catch (error) {
console.error('QR Code generation failed:', error);
return this.createErrorSVG(size);
}
}
// Create QR code data matrix (simplified implementation)
createQRData(text, errorCorrection) {
// For simplicity, we'll create a basic QR code pattern
// This is a minimal implementation - real QR codes are much more complex
const version = this.determineVersion(text.length);
const size = 21 + (version - 1) * 4; // QR code size formula
// Initialize matrix
const matrix = Array(size).fill().map(() => Array(size).fill(false));
// Add finder patterns (corners)
this.addFinderPatterns(matrix);
// Add timing patterns
this.addTimingPatterns(matrix);
// Add data (simplified - just create a pattern based on text)
this.addDataPattern(matrix, text);
return matrix;
}
determineVersion(length) {
// Simplified version determination
if (length <= 25) return 1;
if (length <= 47) return 2;
if (length <= 77) return 3;
return 4; // Max we'll support in this simple implementation
}
addFinderPatterns(matrix) {
const size = matrix.length;
const pattern = [
[1,1,1,1,1,1,1],
[1,0,0,0,0,0,1],
[1,0,1,1,1,0,1],
[1,0,1,1,1,0,1],
[1,0,1,1,1,0,1],
[1,0,0,0,0,0,1],
[1,1,1,1,1,1,1]
];
// Top-left
this.placePattern(matrix, 0, 0, pattern);
// Top-right
this.placePattern(matrix, 0, size - 7, pattern);
// Bottom-left
this.placePattern(matrix, size - 7, 0, pattern);
}
addTimingPatterns(matrix) {
const size = matrix.length;
// Horizontal timing pattern
for (let i = 8; i < size - 8; i++) {
matrix[6][i] = i % 2 === 0;
}
// Vertical timing pattern
for (let i = 8; i < size - 8; i++) {
matrix[i][6] = i % 2 === 0;
}
}
addDataPattern(matrix, text) {
const size = matrix.length;
// Simple data pattern based on text hash
let hash = 0;
for (let i = 0; i < text.length; i++) {
hash = ((hash << 5) - hash + text.charCodeAt(i)) & 0xffffffff;
}
// Fill available spaces with pattern based on hash
for (let row = 0; row < size; row++) {
for (let col = 0; col < size; col++) {
if (!this.isReserved(row, col, size)) {
matrix[row][col] = ((hash >> ((row + col) % 32)) & 1) === 1;
}
}
}
}
placePattern(matrix, startRow, startCol, pattern) {
for (let row = 0; row < pattern.length; row++) {
for (let col = 0; col < pattern[row].length; col++) {
matrix[startRow + row][startCol + col] = pattern[row][col] === 1;
}
}
}
isReserved(row, col, size) {
// Check if position is reserved for finder patterns, timing patterns, etc.
// Finder patterns
if ((row < 9 && col < 9) || // Top-left
(row < 9 && col >= size - 8) || // Top-right
(row >= size - 8 && col < 9)) { // Bottom-left
return true;
}
// Timing patterns
if (row === 6 || col === 6) {
return true;
}
return false;
}
createErrorSVG(size) {
return `<svg width="${size}" height="${size}" viewBox="0 0 ${size} ${size}" xmlns="http://www.w3.org/2000/svg">
<rect width="${size}" height="${size}" fill="#f3f4f6"/>
<text x="${size/2}" y="${size/2-10}" text-anchor="middle" font-family="Arial, sans-serif" font-size="14" fill="#6b7280">QR Code</text>
<text x="${size/2}" y="${size/2+10}" text-anchor="middle" font-family="Arial, sans-serif" font-size="14" fill="#6b7280">Error</text>
</svg>`;
}
}
// Global function for easy access
window.generateQRCode = function(text, containerId, options = {}) {
const generator = new QRCodeGenerator();
const container = document.getElementById(containerId);
if (!container) {
console.error('Container not found:', containerId);
return;
}
const svg = generator.generateSVG(text, options);
container.innerHTML = svg;
};

View File

@@ -13,3 +13,16 @@
/* Import pages */
@import "pages/home";
/* QR Code Styles */
.qr-code-container {
@apply flex items-center justify-center;
}
.qr-code-container svg {
max-width: 100% !important;
max-height: 100% !important;
width: 208px !important;
height: 208px !important;
display: block !important;
}

View File

@@ -0,0 +1,141 @@
/* PDF Styles for Ticket Generation */
body {
font-family: Helvetica, Arial, sans-serif;
font-size: 12px;
color: #000000;
margin: 0;
padding: 20px;
background-color: #ffffff;
}
.ticket-container {
max-width: 350px;
margin: 0 auto;
padding: 20px;
border: 1px solid #e5e7eb;
border-radius: 10px;
background-color: #ffffff;
}
/* Header */
.header {
text-align: center;
margin-bottom: 10px;
}
.header h1 {
color: #2D1B69;
font-size: 24px;
font-weight: bold;
margin: 0;
}
/* Event name */
.event-name {
text-align: center;
margin-bottom: 20px;
}
.event-name h2 {
color: #000000;
font-size: 18px;
font-weight: bold;
margin: 0;
}
/* Ticket info box */
.ticket-info-box {
background-color: #F9FAFB;
border: 1px solid #E5E7EB;
border-radius: 10px;
padding: 15px;
margin-bottom: 20px;
}
.info-row {
margin-bottom: 8px;
}
.info-row:last-child {
margin-bottom: 0;
}
.info-label {
font-weight: bold;
color: #000000;
display: inline-block;
width: 100px;
}
.info-value {
display: inline-block;
color: #000000;
}
/* Venue information */
.venue-info {
margin-bottom: 20px;
}
.venue-info h3 {
color: #374151;
font-size: 14px;
font-weight: bold;
margin: 0 0 8px 0;
}
.venue-details {
font-size: 11px;
}
.venue-name {
font-weight: bold;
margin-bottom: 4px;
}
.venue-address {
color: #000000;
}
/* QR Code */
.qr-code-section {
text-align: center;
margin-bottom: 15px;
}
.qr-code-section h3 {
color: #000000;
font-size: 14px;
font-weight: bold;
margin: 0 0 10px 0;
}
.qr-code-container {
text-align: center;
margin: 0 auto 10px auto;
width: 120px;
height: 120px;
}
.qr-code-text {
font-size: 8px;
color: #6B7280;
}
/* Footer */
.footer {
border-top: 1px solid #E5E7EB;
padding-top: 15px;
text-align: center;
font-size: 8px;
color: #6B7280;
}
.footer p {
margin: 0 0 5px 0;
}
.generated-date {
margin-top: 5px;
}

View File

@@ -5,9 +5,6 @@ class ApplicationController < ActionController::Base
# Ensures that all non-GET requests include a valid authenticity token
protect_from_forgery with: :exception
# Redirect authenticated users to onboarding if not completed
before_action :require_onboarding_completion
# Restrict access to modern browsers only
# Requires browsers to support modern web standards:
# - WebP images for better compression
@@ -17,29 +14,4 @@ class ApplicationController < ActionController::Base
# - CSS nesting and :has() pseudo-class
# allow_browser versions: :modern
# allow_browser versions: { safari: 16.4, firefox: 121, ie: false }
private
def require_onboarding_completion
# Skip onboarding check for these paths
return if skip_onboarding_check?
# Only apply to signed-in users
if user_signed_in? && current_user.needs_onboarding?
redirect_to onboarding_path unless request.path == onboarding_path
end
end
def skip_onboarding_check?
# Skip for devise controllers (login, signup, password reset, etc.)
devise_controller? ||
# Skip for onboarding controller itself
controller_name == "onboarding" ||
# Skip for API endpoints
controller_name.start_with?("api/") ||
# Skip for health checks
controller_name == "rails/health" ||
# Skip for home page (when not signed in)
(controller_name == "pages" && action_name == "home")
end
end

View File

@@ -1,38 +0,0 @@
class OnboardingController < ApplicationController
before_action :authenticate_user!
before_action :redirect_if_onboarding_complete, except: [:complete]
def index
# Display the onboarding form
end
def complete
if onboarding_params_valid?
current_user.update!(onboarding_params)
current_user.complete_onboarding!
flash[:notice] = "Bienvenue sur AperoNight ! Votre profil a été configuré avec succès."
redirect_to dashboard_path
else
flash.now[:alert] = "Veuillez remplir tous les champs requis."
render :index
end
end
private
def onboarding_params
params.require(:user).permit(:first_name, :last_name)
end
def onboarding_params_valid?
onboarding_params[:first_name].present? &&
onboarding_params[:last_name].present?
end
def redirect_if_onboarding_complete
if current_user&.onboarding_completed?
redirect_to dashboard_path
end
end
end

View File

@@ -4,7 +4,7 @@
# Orders group multiple tickets together for better transaction management
class OrdersController < ApplicationController
before_action :authenticate_user!
before_action :set_order, only: [ :show, :checkout, :retry_payment, :increment_payment_attempt, :invoice ]
before_action :set_order, only: [ :show, :checkout, :retry_payment, :increment_payment_attempt ]
before_action :set_event, only: [ :new, :create ]
# Display new order form with name collection
@@ -97,15 +97,9 @@ class OrdersController < ApplicationController
redirect_to event_order_new_path(@event.slug, @event.id)
end
# Display all user orders
def index
@orders = current_user.orders.includes(:event, tickets: :ticket_type)
.where(status: [ "paid", "completed" ])
.order(created_at: :desc)
.page(params[:page])
end
# Display order summary
#
#
def show
@tickets = @order.tickets.includes(:ticket_type)
end
@@ -155,26 +149,6 @@ class OrdersController < ApplicationController
redirect_to checkout_order_path(@order)
end
# Display invoice for an order
def invoice
unless @order.status == "paid" || @order.status == "completed"
redirect_to order_path(@order), alert: "La facture n'est disponible qu'après le paiement de la commande"
return
end
@tickets = @order.tickets.includes(:ticket_type)
# Get the Stripe invoice if it exists
begin
@stripe_invoice_id = @order.create_stripe_invoice!
@stripe_invoice_pdf_url = @order.stripe_invoice_pdf_url if @stripe_invoice_id
rescue => e
Rails.logger.error "Failed to retrieve or create Stripe invoice for order #{@order.id}: #{e.message}"
@stripe_invoice_id = nil
@stripe_invoice_pdf_url = nil
end
end
# Handle successful payment
def payment_success
session_id = params[:session_id]
@@ -214,8 +188,15 @@ class OrdersController < ApplicationController
# Don't fail the payment process due to job scheduling issues
end
# Email confirmation is handled by the order model's mark_as_paid! method
# to avoid duplicate emails
# Send confirmation emails
@order.tickets.each do |ticket|
begin
TicketMailer.purchase_confirmation(ticket).deliver_now
rescue => e
Rails.logger.error "Failed to send confirmation email for ticket #{ticket.id}: #{e.message}"
# Don't fail the entire payment process due to email/PDF generation issues
end
end
# Clear session data
session.delete(:pending_cart)
@@ -288,19 +269,6 @@ class OrdersController < ApplicationController
}
end
# Add service fee as a separate line item
line_items << {
price_data: {
currency: "eur",
product_data: {
name: "Frais de service",
description: "Frais de traitement de la commande"
},
unit_amount: 100 # 1€ in cents
},
quantity: 1
}
Stripe::Checkout::Session.create(
payment_method_types: [ "card" ],
line_items: line_items,

View File

@@ -17,28 +17,30 @@ class PagesController < ApplicationController
# User dashboard showing personalized content
# Accessible only to authenticated users
def dashboard
# User's orders with associated data
@user_orders = current_user.orders.includes(:event, tickets: :ticket_type)
.where(status: [ "paid", "completed" ])
.order(created_at: :desc)
.limit(10)
# Metrics for dashboard cards
@booked_events = current_user.orders.joins(tickets: { ticket_type: :event })
.where(events: { state: :published })
.where(orders: { status: [ "paid", "completed" ] })
.sum("1")
@events_today = Event.published.where("DATE(start_time) = ?", Date.current).count
@events_tomorrow = Event.published.where("DATE(start_time) = ?", Date.current + 1).count
@upcoming_events = Event.published.upcoming.count
# User's booked events
@user_booked_events = Event.joins(ticket_types: { tickets: :order })
.where(orders: { user: current_user }, tickets: { status: "active" })
.distinct
.limit(5)
# Draft orders that can be retried
@draft_orders = current_user.orders.includes(tickets: [ :ticket_type, :event ])
.can_retry_payment
.order(:expires_at)
# Simplified upcoming events preview - only show if user has orders
if @user_orders.any?
ordered_event_ids = @user_orders.map(&:event).map(&:id)
@upcoming_preview_events = Event.published
.upcoming
.where.not(id: ordered_event_ids)
.order(start_time: :asc)
.limit(6)
else
@upcoming_preview_events = []
end
# Events sections
@today_events = Event.published.where("DATE(start_time) = ?", Date.current).order(start_time: :asc)
@tomorrow_events = Event.published.where("DATE(start_time) = ?", Date.current + 1).order(start_time: :asc)
@other_events = Event.published.upcoming.where.not("DATE(start_time) IN (?)", [ Date.current, Date.current + 1 ]).order(start_time: :asc).page(params[:page])
end
# Events page showing all published events with pagination

View File

@@ -3,10 +3,9 @@
# This controller now primarily handles legacy redirects and backward compatibility
# Most ticket creation functionality has been moved to OrdersController
class TicketsController < ApplicationController
before_action :authenticate_user!, only: [ :payment_success, :payment_cancel, :show, :download ]
before_action :authenticate_user!, only: [ :payment_success, :payment_cancel, :show, :ticket_view, :download_ticket ]
before_action :set_event, only: [ :checkout, :retry_payment ]
# Redirect to order-based checkout
def checkout
# Check for draft order
@@ -50,18 +49,28 @@ class TicketsController < ApplicationController
# Display ticket details
def show
# Find ticket by qr code id
@ticket = Ticket.joins(order: :user).includes(:event, :ticket_type, order: :user)
.find_by(tickets: { qr_code: params[:qr_code] })
@ticket = Ticket.joins(order: :user).includes(:event, :ticket_type, order: :user).find_by(
tickets: { id: params[:ticket_id] },
orders: { user_id: current_user.id }
)
@event = @ticket.event
rescue ActiveRecord::RecordNotFound
redirect_to dashboard_path, alert: "Billet non trouvé"
end
# Display ticket in PDF-like format
def ticket_view
@ticket = Ticket.joins(order: :user).includes(:event, :ticket_type, order: :user).find_by(
tickets: { id: params[:ticket_id] },
orders: { user_id: current_user.id }
)
if @ticket.nil?
redirect_to dashboard_path, alert: "Billet non trouvé"
redirect_to dashboard_path, alert: "Billet non trouvé ou vous n'avez pas l'autorisation d'accéder à ce billet"
return
end
@event = @ticket.event
@order = @ticket.order
rescue ActiveRecord::RecordNotFound
redirect_to dashboard_path, alert: "Billet non trouvé"
end
@@ -69,30 +78,104 @@ class TicketsController < ApplicationController
# Download PDF ticket - only accessible by ticket owner
# User must be authenticated to download ticket
# TODO: change ID to an unique identifier (UUID)
def download
# Find ticket by qr code id
@ticket = Ticket.joins(order: :user).includes(:event, :ticket_type, order: :user)
.find_by(tickets: { qr_code: params[:qr_code] })
def download_ticket
# Find ticket and ensure it belongs to current user
@ticket = Ticket.joins(order: :user).includes(:event, :ticket_type, order: :user).find_by(
tickets: { id: params[:ticket_id] },
orders: { user_id: current_user.id }
)
if @ticket.nil?
redirect_to dashboard_path, alert: "Billet non trouvé ou vous n'avez pas l'autorisation d'accéder à ce billet"
return
end
# Generate PDF
pdf_content = @ticket.to_pdf
# Generate PDF using Grover
begin
Rails.logger.info "Starting PDF generation for ticket ID: #{@ticket.id}"
# Send PDF as download
send_data pdf_content,
filename: "ticket_#{@ticket.id}_#{@ticket.event.name.parameterize}.pdf",
type: "application/pdf",
disposition: "attachment"
rescue ActiveRecord::RecordNotFound
# Render the HTML template
html = render_to_string(
partial: "tickets/pdf_ticket",
layout: false,
locals: { ticket: @ticket }
)
Rails.logger.info "HTML template rendered successfully, length: #{html.length}"
# Try to load and use Grover
begin
Rails.logger.info "Attempting to load Grover gem"
# Try different approaches to load grover
begin
require "bundler"
Bundler.require(:default, Rails.env)
Rails.logger.info "Bundler required gems successfully"
rescue => bundler_error
Rails.logger.warn "Bundler require failed: #{bundler_error.message}"
end
# Direct path approach using bundle show
grover_gem_path = `bundle show grover`.strip
grover_path = File.join(grover_gem_path, "lib", "grover")
if File.exist?(grover_path + ".rb")
Rails.logger.info "Loading Grover from direct path: #{grover_path}"
require grover_path
else
Rails.logger.error "Grover not found at path: #{grover_path}"
raise LoadError, "Grover gem not available at expected path"
end
Rails.logger.info "Creating Grover instance with options"
grover = Grover.new(html,
format: "A6",
margin: {
top: "10mm",
bottom: "10mm",
left: "10mm",
right: "10mm"
},
prefer_css_page_size: true,
emulate_media: "print",
cache: false,
launch_args: [ "--no-sandbox", "--disable-setuid-sandbox" ] # For better compatibility
)
Rails.logger.info "Grover instance created successfully"
pdf_content = grover.to_pdf
Rails.logger.info "PDF generated successfully, length: #{pdf_content.length}"
# Send PDF as download
send_data pdf_content,
filename: "ticket_#{@ticket.id}_#{@ticket.event.name.parameterize}.pdf",
type: "application/pdf",
disposition: "attachment"
rescue LoadError => grover_error
Rails.logger.error "Failed to load Grover: #{grover_error.message}"
# Fallback: return HTML instead of PDF
send_data html,
filename: "ticket_#{@ticket.id}_#{@ticket.event.name.parameterize}.html",
type: "text/html",
disposition: "attachment"
end
rescue => e
Rails.logger.error "Error generating ticket PDF with Grover:"
Rails.logger.error "Message: #{e.message}"
Rails.logger.error "Backtrace: #{e.backtrace.join("\n")}"
redirect_to dashboard_path, alert: "Erreur lors de la génération du billet"
end
rescue ActiveRecord::RecordNotFound => e
Rails.logger.error "ActiveRecord::RecordNotFound error: #{e.message}"
redirect_to dashboard_path, alert: "Billet non trouvé"
rescue => e
Rails.logger.error "Error generating ticket PDF: #{e.message}"
Rails.logger.error "Unexpected error in download_ticket action:"
Rails.logger.error "Message: #{e.message}"
Rails.logger.error "Backtrace: #{e.backtrace.join("\n")}"
redirect_to dashboard_path, alert: "Erreur lors de la génération du billet"
end
private
def set_event

View File

@@ -1,2 +0,0 @@
module OnboardingHelper
end

View File

@@ -18,6 +18,3 @@ application.register("ticket-selection", TicketSelectionController);
import HeaderController from "./header_controller";
application.register("header", HeaderController);
import QrCodeController from "./qr_code_controller";
application.register("qr-code", QrCodeController);

View File

@@ -1,56 +0,0 @@
// QR Code generator controller using qrcode npm package
import { Controller } from "@hotwired/stimulus"
import QRCode from "qrcode"
export default class extends Controller {
static values = { data: String }
static targets = ["container", "loading"]
connect() {
this.generateQRCode()
}
async generateQRCode() {
try {
// Hide loading indicator
if (this.hasLoadingTarget) {
this.loadingTarget.style.display = 'none'
}
// Create canvas element
const canvas = document.createElement('canvas')
// Generate QR code using qrcode library
await QRCode.toCanvas(canvas, this.dataValue, {
width: 128,
height: 128,
margin: 1,
color: {
dark: '#000000',
light: '#FFFFFF'
}
})
// Clear container and add QR code
this.containerTarget.innerHTML = ''
this.containerTarget.appendChild(canvas)
console.log('QR code generated successfully')
} catch (error) {
console.error('Error generating QR code:', error)
this.showFallback()
}
}
showFallback() {
this.containerTarget.innerHTML = `
<div class="w-32 h-32 bg-gray-100 rounded flex items-center justify-center text-gray-500 text-xs border-2 border-dashed border-gray-300">
<div class="text-center">
<div class="text-lg mb-1">📱</div>
<div>QR Code</div>
<div class="font-mono text-xs mt-1 break-all px-2">${this.dataValue}</div>
</div>
</div>
`
}
}

View File

@@ -1,19 +0,0 @@
class EventReminderJob < ApplicationJob
queue_as :default
def perform(event_id, days_before)
event = Event.find(event_id)
# Find all users with active tickets for this event
users_with_tickets = User.joins(orders: { tickets: :ticket_type })
.where(ticket_types: { event: event })
.where(tickets: { status: "active" })
.distinct
users_with_tickets.find_each do |user|
TicketMailer.event_reminder(user, event, days_before).deliver_now
rescue StandardError => e
Rails.logger.error "Failed to send event reminder to user #{user.id} for event #{event.id}: #{e.message}"
end
end
end

View File

@@ -1,44 +0,0 @@
class EventReminderSchedulerJob < ApplicationJob
queue_as :default
def perform
schedule_weekly_reminders
schedule_daily_reminders
schedule_day_of_reminders
end
private
def schedule_weekly_reminders
# Find events starting in exactly 7 days
target_date = 7.days.from_now.beginning_of_day
events = Event.published
.where(start_time: target_date..(target_date + 1.day))
events.find_each do |event|
EventReminderJob.perform_later(event.id, 7)
end
end
def schedule_daily_reminders
# Find events starting in exactly 1 day (tomorrow)
target_date = 1.day.from_now.beginning_of_day
events = Event.published
.where(start_time: target_date..(target_date + 1.day))
events.find_each do |event|
EventReminderJob.perform_later(event.id, 1)
end
end
def schedule_day_of_reminders
# Find events starting today
target_date = Time.current.beginning_of_day
events = Event.published
.where(start_time: target_date..(target_date + 1.day))
events.find_each do |event|
EventReminderJob.perform_later(event.id, 0)
end
end
end

View File

@@ -1,4 +1,4 @@
class ApplicationMailer < ActionMailer::Base
default from: ENV.fetch("MAILER_FROM_EMAIL", "no-reply@aperonight.fr")
default from: "from@example.com"
layout "mailer"
end

View File

@@ -1,30 +1,5 @@
class TicketMailer < ApplicationMailer
def purchase_confirmation_order(order)
@order = order
@user = order.user
@event = order.event
@tickets = order.tickets
# Generate PDF attachments for all tickets
@tickets.each do |ticket|
begin
pdf = ticket.to_pdf
attachments["ticket-#{@event.name.parameterize}-#{ticket.qr_code[0..7]}.pdf"] = {
mime_type: "application/pdf",
content: pdf
}
rescue StandardError => e
Rails.logger.error "Failed to generate PDF for ticket #{ticket.id}: #{e.message}"
# Continue without PDF attachment rather than failing the entire email
end
end
mail(
to: @user.email,
subject: "Confirmation d'achat - #{@event.name}",
template_name: "purchase_confirmation"
)
end
default from: "notifications@aperonight.com"
def purchase_confirmation(ticket)
@ticket = ticket
@@ -32,49 +7,15 @@ class TicketMailer < ApplicationMailer
@event = ticket.event
# Generate PDF attachment
begin
pdf = @ticket.to_pdf
attachments["ticket-#{@event.name.parameterize}-#{@ticket.qr_code[0..7]}.pdf"] = {
mime_type: "application/pdf",
content: pdf
}
rescue StandardError => e
Rails.logger.error "Failed to generate PDF for ticket #{@ticket.id}: #{e.message}"
# Continue without PDF attachment rather than failing the entire email
end
pdf = @ticket.to_pdf
attachments["ticket-#{@event.name.parameterize}-#{@ticket.qr_code[0..7]}.pdf"] = {
mime_type: "application/pdf",
content: pdf
}
mail(
to: @user.email,
subject: "Confirmation d'achat - #{@event.name}"
)
end
def event_reminder(user, event, days_before)
@user = user
@event = event
@days_before = days_before
# Get user's tickets for this event
@tickets = Ticket.joins(:order, :ticket_type)
.where(orders: { user: @user }, ticket_types: { event: @event }, status: "active")
return if @tickets.empty?
subject = case days_before
when 7
"Rappel : #{@event.name} dans une semaine"
when 1
"Rappel : #{@event.name} demain"
when 0
"C'est aujourd'hui : #{@event.name}"
else
"Rappel : #{@event.name} dans #{days_before} jours"
end
mail(
to: @user.email,
subject: subject,
template_name: "event_reminder"
)
end
end

View File

@@ -76,23 +76,11 @@ class Order < ApplicationRecord
update!(status: "paid")
tickets.update_all(status: "active")
end
# Send purchase confirmation email outside the transaction
# so that payment completion isn't affected by email failures
begin
TicketMailer.purchase_confirmation_order(self).deliver_now
rescue StandardError => e
Rails.logger.error "Failed to send purchase confirmation email for order #{id}: #{e.message}"
Rails.logger.error e.backtrace.join("\n")
# Don't re-raise the error - payment should still succeed
end
end
# Calculate total from tickets plus 1€ service fee
# Calculate total from tickets
def calculate_total!
ticket_total = tickets.sum(:price_cents)
fee_cents = 100 # 1€ in cents
update!(total_amount_cents: ticket_total + fee_cents)
update!(total_amount_cents: tickets.sum(:price_cents))
end
# Create Stripe invoice for accounting records

View File

@@ -27,6 +27,29 @@ class Ticket < ApplicationRecord
TicketPdfGenerator.new(self).generate
end
# Generate QR code data for ticket validation
def to_qr_data
{
ticket_id: id,
qr_code: qr_code,
event_id: event&.id,
user_id: user&.id
}.compact.to_json
end
# Generate QR code as SVG
def generate_qr_svg
require "rqrcode"
qrcode = RQRCode::QRCode.new(to_qr_data)
qrcode.as_svg(
offset: 0,
color: "000",
shape_rendering: "crispEdges",
module_size: 4,
standalone: true
)
end
# Price in euros (formatted)
def price_euros
price_cents / 100.0
@@ -70,6 +93,7 @@ class Ticket < ApplicationRecord
self.qr_code = "#{id || 'temp'}-#{Time.current.to_i}-#{SecureRandom.hex(4)}"
end
def draft?
status == "draft"
end

View File

@@ -29,15 +29,6 @@ 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 }
# Onboarding methods
def needs_onboarding?
!onboarding_completed?
end
def complete_onboarding!
update!(onboarding_completed: true)
end
# Authorization methods
def can_manage_events?
# For now, all authenticated users can manage events

View File

@@ -147,7 +147,6 @@ class StripeInvoiceService
end
def add_line_items_to_invoice(customer, invoice)
# Add ticket line items
@order.tickets.group_by(&:ticket_type).each do |ticket_type, tickets|
quantity = tickets.count
@@ -165,20 +164,6 @@ class StripeInvoiceService
}
})
end
# Add service fee line item
service_fee_cents = 100 # 1€ service fee
Stripe::InvoiceItem.create({
customer: customer.id,
invoice: invoice.id,
amount: service_fee_cents,
currency: "eur",
description: "Frais de service - Frais de traitement de la commande",
metadata: {
item_type: "service_fee",
amount_cents: service_fee_cents
}
})
end
def build_line_item_description(ticket_type, tickets)

View File

@@ -1,297 +0,0 @@
require "prawn"
require "prawn/qrcode"
require "rqrcode"
# Service de génération de billets PDF utilisant Prawn
#
# Génère des billets PDF simples et compacts avec codes QR pour la validation d'entrée
# Design propre et minimaliste qui tient sur une seule page
class TicketPdfGenerator
# Suppress Prawn's internationalization warning for built-in fonts
Prawn::Fonts::AFM.hide_m17n_warning = true
attr_reader :ticket
def initialize(ticket)
@ticket = ticket
end
def generate
Prawn::Document.new(page_size: [ 350, 600 ], margin: 20) do |pdf|
# Header
pdf.fill_color "2D1B69"
pdf.font "Helvetica", style: :bold, size: 24
pdf.text ENV.fetch("APP_NAME", "Aperonight"), align: :center
pdf.move_down 10
# Event name
pdf.fill_color "000000"
pdf.font "Helvetica", style: :bold, size: 18
pdf.text ticket.event.name, align: :center
pdf.move_down 10
# Ticket info box
pdf.stroke_color "E5E7EB"
pdf.fill_color "F9FAFB"
pdf.rounded_rectangle [ 0, pdf.cursor ], 310, 150, 10
pdf.fill_and_stroke
pdf.move_down 10
pdf.fill_color "000000"
pdf.font "Helvetica", size: 12
# Customer name
pdf.indent 10 do
pdf.text "Titulaire du billet :", style: :bold
pdf.text "#{ticket.first_name} #{ticket.last_name}"
end
pdf.move_down 8
# Ticket details
pdf.indent 10 do
pdf.text "Type de billet :", style: :bold
pdf.text ticket.ticket_type.name
end
pdf.move_down 8
pdf.indent 10 do
pdf.text "Prix :", style: :bold
pdf.text "#{ticket.price_euros}"
end
pdf.move_down 8
pdf.indent 10 do
pdf.text "Date et heure :", style: :bold
pdf.text ticket.event.start_time.strftime("%d %B %Y à %H:%M")
end
pdf.move_down 20
# Informations sur le lieu
pdf.fill_color "374151"
pdf.font "Helvetica", style: :bold, size: 14
pdf.text "Informations sur le lieu"
pdf.move_down 8
pdf.font "Helvetica", size: 11
pdf.text ticket.event.venue_name, style: :bold
pdf.text ticket.event.venue_address
pdf.move_down 20
# Code QR
pdf.fill_color "000000"
pdf.font "Helvetica", style: :bold, size: 14
pdf.text "Code QR", align: :center
pdf.move_down 10
# Ensure all required data is present before generating QR code
if ticket.qr_code.blank?
raise "Ticket QR code is missing"
end
# Build QR code data with safe association loading
qr_code_data = build_qr_code_data(ticket)
# Validate QR code data before creating QR code
if qr_code_data.blank? || qr_code_data == "{}"
Rails.logger.error "QR code data is empty: ticket_id=#{ticket.id}, qr_code=#{ticket.qr_code}, event_id=#{ticket.ticket_type&.event_id}, user_id=#{ticket.order&.user_id}"
raise "QR code data is empty or invalid"
end
# Ensure qr_code_data is a proper string for QR code generation
unless qr_code_data.is_a?(String) && qr_code_data.length > 2
Rails.logger.error "QR code data is not a valid string: #{qr_code_data.inspect} (class: #{qr_code_data.class})"
raise "QR code data must be a valid string"
end
# Generate QR code - prawn-qrcode expects the data string directly
pdf.print_qr_code(qr_code_data, extent: 120, align: :center)
pdf.move_down 15
# QR code text
pdf.font "Helvetica", size: 8
pdf.fill_color "6B7280"
pdf.text "#{ticket.qr_code}", align: :center
# Ticket ID
pdf.font "Helvetica", size: 8
pdf.fill_color "6B7280"
pdf.text "ID du billet : #{ticket.id}", align: :center
# Footer
pdf.move_down 30
pdf.stroke_color "E5E7EB"
pdf.horizontal_line 0, 310
pdf.move_down 6
pdf.font "Helvetica", size: 8
pdf.fill_color "6B7280"
pdf.text "Ce billet est valable pour une seule entrée.", align: :center
pdf.text "Présentez ce billet à l'entrée du lieu.", align: :center
pdf.move_down 5
pdf.text "Généré le #{Time.current.strftime('%d %B %Y à %H:%M')}", align: :center
end.render
end
private
def create_simple_header(pdf)
# Nom de la marque
pdf.fill_color "6366F1"
pdf.font "Helvetica", style: :bold, size: 24
pdf.text "AperoNight", align: :center
pdf.move_down 5
pdf.font "Helvetica", size: 10
pdf.fill_color "64748B"
pdf.text "Billet d'entree", align: :center
pdf.move_down 20
# Simple divider line
pdf.stroke_color "E5E7EB"
pdf.horizontal_line 0, pdf.bounds.width
pdf.move_down 20
end
def create_ticket_info(pdf)
# Nom de l'événement - proéminent
pdf.fill_color "1F2937"
pdf.font "Helvetica", style: :bold, size: 18
pdf.text ticket.event.name, align: :center
pdf.move_down 15
# Two-column layout for ticket details
pdf.bounding_box([ 0, pdf.cursor ], width: pdf.bounds.width, height: 120) do
# Left column
pdf.bounding_box([ 0, pdf.cursor ], width: pdf.bounds.width / 2 - 20, height: 120) do
create_info_item(pdf, "Date", ticket.event.start_time.strftime("%d %B %Y"))
create_info_item(pdf, "Heure", ticket.event.start_time.strftime("%H:%M"))
create_info_item(pdf, "Lieu", ticket.event.venue_name)
end
# Right column
pdf.bounding_box([ pdf.bounds.width / 2 + 20, pdf.cursor ], width: pdf.bounds.width / 2 - 20, height: 120) do
create_info_item(pdf, "Type", ticket.ticket_type.name)
create_info_item(pdf, "Prix", "#{sprintf('%.2f', ticket.price_euros)}")
create_info_item(pdf, "Titulaire", "#{ticket.first_name} #{ticket.last_name}")
end
end
pdf.move_down 30
end
def create_info_item(pdf, label, value)
pdf.font "Helvetica", style: :bold, size: 9
pdf.fill_color "64748B"
pdf.text label.upcase
pdf.move_down 2
pdf.font "Helvetica", size: 11
pdf.fill_color "1F2937"
pdf.text value
pdf.move_down 12
end
def create_qr_section(pdf)
# Center the QR code horizontally
qr_size = 120
x_position = (pdf.bounds.width - qr_size) / 2
pdf.bounding_box([ x_position, pdf.cursor ], width: qr_size, height: qr_size + 40) do
# QR Code title
pdf.font "Helvetica", style: :bold, size: 12
pdf.fill_color "1F2937"
pdf.text "Code d'entree", align: :center
pdf.move_down 10
# Generate QR code
generate_simple_qr_code(pdf, qr_size)
pdf.move_down 10
# QR code ID
pdf.font "Helvetica", size: 8
pdf.fill_color "64748B"
pdf.text "ID: #{ticket.qr_code[0..15]}...", align: :center
end
pdf.move_down 40
end
def generate_simple_qr_code(pdf, size)
# Ensure all required data is present before generating QR code
if ticket.qr_code.blank?
raise "Ticket QR code is missing"
end
# Build QR code data with safe association loading
qr_code_data = build_qr_code_data(ticket)
# Validate QR code data before creating QR code
if qr_code_data.blank? || qr_code_data == "{}"
Rails.logger.error "QR code data is empty: ticket_id=#{ticket.id}, qr_code=#{ticket.qr_code}, event_id=#{ticket.ticket_type&.event_id}, user_id=#{ticket.order&.user_id}"
raise "QR code data is empty or invalid"
end
# Ensure qr_code_data is a proper string for QR code generation
unless qr_code_data.is_a?(String) && qr_code_data.length > 2
Rails.logger.error "QR code data is not a valid string: #{qr_code_data.inspect} (class: #{qr_code_data.class})"
raise "QR code data must be a valid string"
end
# Generate QR code
pdf.print_qr_code(qr_code_data, extent: size, align: :center)
end
def create_simple_footer(pdf)
# Security notice
pdf.font "Helvetica", size: 8
pdf.fill_color "64748B"
pdf.text "Ce billet est valable pour une seule entree.", align: :center
pdf.text "Presentez ce code QR a l'entree de l'evenement.", align: :center
pdf.move_down 10
# Divider line
pdf.stroke_color "E5E7EB"
pdf.horizontal_line 0, pdf.bounds.width
pdf.move_down 5
# Generation timestamp
pdf.font "Helvetica", size: 7
pdf.fill_color "9CA3AF"
timestamp = "Genere le #{Time.current.strftime('%d/%m/%Y a %H:%M')}"
pdf.text timestamp, align: :center
end
def build_qr_code_data(ticket)
# Try multiple approaches to get valid QR code data
begin
# Primary approach: full JSON with all data
data = {
ticket_id: ticket.id,
qr_code: ticket.qr_code,
event_id: ticket.ticket_type&.event_id,
user_id: ticket.order&.user_id
}.compact
# Ensure we have the minimum required data
if data[:ticket_id] && data[:qr_code]
return data.to_json
end
rescue StandardError => e
Rails.logger.warn "Failed to build complex QR data: #{e.message}"
end
# Fallback approach: just use the ticket's QR code string
begin
return ticket.qr_code.to_s if ticket.qr_code.present?
rescue StandardError => e
Rails.logger.warn "Failed to use ticket QR code: #{e.message}"
end
# Final fallback: simple ticket identifier
"TICKET-#{ticket.id}"
end
end

View File

@@ -37,12 +37,10 @@
<div class="font-medium"><%= current_user.first_name || current_user.email %></div>
<div class="text-gray-500"><%= current_user.email %></div>
</div>
<%= link_to "Profile", edit_user_registration_path,
class: "block px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 transition-colors duration-200" %>
<%= link_to "Reservations", "#",
class: "block px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 transition-colors duration-200" %>
<div class="border-t border-gray-100">
<%= link_to "Sécurité", edit_user_registration_path,
class: "block px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 transition-colors duration-200" %>
</div>
<div class="border-t border-gray-100">
<%= link_to "Sign out", destroy_user_session_path,
data: { controller: "logout", action: "click->logout#signOut", logout_url_value: destroy_user_session_path, login_url_value: new_user_session_path, turbo: false },
@@ -88,11 +86,11 @@
</div>
</div>
<div class="px-2 space-y-1">
<%= link_to "Réservations", "#",
<%= link_to t("header.profile"), edit_user_registration_path,
class: "block px-3 py-2 rounded-md text-base font-medium text-gray-100 hover:text-purple-200 hover:bg-purple-700" %>
<%= link_to "Sécurité", edit_user_registration_path,
<%= link_to t("header.reservations"), "#",
class: "block px-3 py-2 rounded-md text-base font-medium text-gray-100 hover:text-purple-200 hover:bg-purple-700" %>
<%= link_to "Déconnexion", destroy_user_session_path,
<%= link_to t("header.logout"), destroy_user_session_path,
data: { controller: "logout", action: "click->logout#signOut", logout_url_value: destroy_user_session_path, login_url_value: new_user_session_path, turbo: false },
class: "block px-3 py-2 rounded-md text-base font-medium text-gray-100 hover:text-purple-200 hover:bg-purple-700" %>
</div>

View File

@@ -8,19 +8,40 @@
</div>
<!-- Breadcrumb -->
<nav class="flex mb-6" aria-label="Breadcrumb">
<ol class="inline-flex items-center space-x-1 md:space-x-2 rounded-lg bg-white px-4 py-2 shadow-sm">
<li class="inline-flex items-center">
<%= link_to "Accueil", root_path, class: "inline-flex items-center text-sm font-medium text-gray-700 hover:text-purple-600" %>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<span class="ml-1 text-sm font-medium text-purple-600 md:ml-2">Événements</span>
</div>
</li>
<nav class="mb-6" aria-label="Breadcrumb">
<ol class="flex items-center space-x-2 text-sm">
<%= link_to root_path, class: "text-gray-500 hover:text-purple-600 transition-colors" do %>
<svg
class="w-4 h-4 inline-block mr-1"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M3 7v10a2 2 0 002 2h14a2 2 0 002-2V9a2 2 0 00-2-2h-6l-2-2H5a2 2 0 00-2 2z"
/>
</svg>
Accueil
<% end %>
<svg
class="w-4 h-4 text-gray-400"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M9 5l7 7-7 7"
/>
</svg>
<%= link_to events_path, class: "text-gray-500 hover:text-purple-600 transition-colors" do %>
Événements
<% end %>
</ol>
</nav>

View File

@@ -84,7 +84,7 @@
</div>
<div class="flex items-center space-x-2">
<%= link_to ticket_download_path(ticket.qr_code, format: :pdf),
<%= link_to download_ticket_path(ticket, format: :pdf),
class: "inline-flex items-center px-4 py-2 bg-gradient-to-r from-purple-600 to-indigo-600 text-white rounded-lg hover:from-purple-700 hover:to-indigo-700 transition-all duration-200 text-sm font-medium shadow-sm" do %>
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"></path>

View File

@@ -0,0 +1,11 @@
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title><%= yield :title %></title>
<%= stylesheet_link_tag "pdf" %>
</head>
<body>
<%= yield %>
</body>
</html>

View File

@@ -1,109 +0,0 @@
<div class="min-h-screen bg-gradient-to-br from-purple-50 to-indigo-50 py-8">
<div class="max-w-2xl mx-auto px-4 sm:px-6 lg:px-8">
<!-- Header -->
<div class="text-center mb-8">
<div class="mx-auto w-20 h-20 bg-purple-100 rounded-full flex items-center justify-center mb-6">
<svg class="w-10 h-10 text-purple-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M14.828 14.828a4 4 0 01-5.656 0M9 10h1m4 0h1m-7 4h12a3 3 0 003-3V7a3 3 0 00-3-3H6a3 3 0 00-3 3v4a3 3 0 003 3z"/>
</svg>
</div>
<h1 class="text-3xl font-bold text-gray-900 mb-3">Bienvenue sur <%= Rails.application.config.app_name %> !</h1>
<p class="text-lg text-gray-600 max-w-lg mx-auto">
Configurons rapidement votre profil pour personnaliser votre expérience.
</p>
</div>
<!-- Onboarding Form -->
<div class="bg-white rounded-2xl shadow-xl p-8">
<%= form_with model: current_user, url: complete_onboarding_path, local: true, method: :post, class: "space-y-6" do |form| %>
<!-- Progress indicator -->
<div class="mb-8">
<div class="flex items-center justify-between text-xs text-gray-500 mb-2">
<span>Étape 1 sur 1</span>
<span>Configuration du profil</span>
</div>
<div class="w-full bg-gray-200 rounded-full h-2">
<div class="bg-purple-600 h-2 rounded-full w-full transition-all duration-300"></div>
</div>
</div>
<!-- Form Fields -->
<div class="space-y-6">
<!-- Personal Information Section -->
<div>
<h2 class="text-xl font-semibold text-gray-900 mb-4 flex items-center">
<svg class="w-5 h-5 mr-2 text-purple-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M16 7a4 4 0 11-8 0 4 4 0 018 0zM12 14a7 7 0 00-7 7h14a7 7 0 00-7-7z"/>
</svg>
Informations personnelles
</h2>
<div class="grid grid-cols-1 md:grid-cols-2 gap-4">
<!-- First Name -->
<div>
<%= form.label :first_name, "Prénom", class: "block text-sm font-medium text-gray-700 mb-2" %>
<%= form.text_field :first_name,
value: current_user.first_name,
class: "w-full px-4 py-3 border border-gray-300 rounded-lg focus:ring-2 focus:ring-purple-500 focus:border-purple-500 transition-colors",
placeholder: "Votre prénom",
required: true %>
</div>
<!-- Last Name -->
<div>
<%= form.label :last_name, "Nom", class: "block text-sm font-medium text-gray-700 mb-2" %>
<%= form.text_field :last_name,
value: current_user.last_name,
class: "w-full px-4 py-3 border border-gray-300 rounded-lg focus:ring-2 focus:ring-purple-500 focus:border-purple-500 transition-colors",
placeholder: "Votre nom de famille",
required: true %>
</div>
</div>
</div>
</div>
<!-- Submit Button -->
<div class="pt-6 border-t border-gray-200">
<div class="space-y-4">
<p class="text-sm text-gray-500">
Vous pourrez modifier ces informations plus tard.
</p>
<%= form.submit "Finaliser mon profil",
class: "w-full px-8 py-3 bg-purple-600 text-white font-semibold rounded-lg hover:bg-purple-700 focus:ring-2 focus:ring-purple-500 focus:ring-offset-2 transition-colors cursor-pointer" %>
</div>
</div>
<% end %>
</div>
<!-- Benefits Preview -->
<div class="mt-8 bg-white rounded-xl shadow-lg p-6">
<h3 class="text-lg font-semibold text-gray-900 mb-4 text-center">
Après la configuration, vous pourrez :
</h3>
<div class="grid grid-cols-1 md:grid-cols-3 gap-4">
<div class="flex items-center p-3 bg-green-50 rounded-lg">
<svg class="w-6 h-6 text-green-600 mr-3" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z"/>
</svg>
<span class="text-sm font-medium text-green-800">Réserver des billets</span>
</div>
<div class="flex items-center p-3 bg-blue-50 rounded-lg">
<svg class="w-6 h-6 text-blue-600 mr-3" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
<span class="text-sm font-medium text-blue-800">Gérer vos commandes</span>
</div>
<div class="flex items-center p-3 bg-purple-50 rounded-lg">
<svg class="w-6 h-6 text-purple-600 mr-3" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 6V4m0 2a2 2 0 100 4m0-4a2 2 0 110 4m-6 8a2 2 0 100-4m0 4a2 2 0 100 4m0-4v2m0-6V4m6 6v10m6-2a2 2 0 100-4m0 4a2 2 0 100 4m0-4v2m0-6V4"/>
</svg>
<span class="text-sm font-medium text-purple-800">Créer des événements</span>
</div>
</div>
</div>
</div>
</div>

View File

@@ -1,35 +1,30 @@
<div class="min-h-screen bg-gradient-to-br from-gray-50 to-gray-100 py-8">
<div class="max-w-4xl mx-auto px-4 sm:px-6 lg:px-8">
<!-- Breadcrumb -->
<nav class="flex mb-6" aria-label="Breadcrumb">
<ol class="inline-flex items-center space-x-1 md:space-x-2 rounded-lg bg-white px-4 py-2 shadow-sm">
<li class="inline-flex items-center">
<%= link_to "Accueil", root_path, class: "inline-flex items-center text-sm font-medium text-gray-700 hover:text-purple-600" %>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<%= link_to "Événements", events_path, class: "ml-1 text-sm font-medium text-gray-700 hover:text-purple-600 md:ml-2" %>
</div>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<%= link_to @order.event.name, event_path(@order.event.slug, @order.event), class: "ml-1 text-sm font-medium text-gray-700 hover:text-purple-600 md:ml-2" %>
</div>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<span class="ml-1 text-sm font-medium text-purple-600 md:ml-2">Commande #<%= @order.id %></span>
</div>
</li>
<nav class="mb-8" aria-label="Breadcrumb">
<ol class="flex items-center space-x-2 text-sm">
<%= link_to root_path, class: "text-gray-500 hover:text-purple-600 transition-colors" do %>
<svg class="w-4 h-4 inline-block mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M3 7v10a2 2 0 002 2h14a2 2 0 002-2V9a2 2 0 00-2-2h-6l-2-2H5a2 2 0 00-2 2z"/>
</svg>
Accueil
<% end %>
<svg class="w-4 h-4 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5l7 7-7 7"/>
</svg>
<%= link_to events_path, class: "text-gray-500 hover:text-purple-600 transition-colors" do %>
Événements
<% end %>
<svg class="w-4 h-4 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5l7 7-7 7"/>
</svg>
<%= link_to event_path(@order.event.slug, @order.event), class: "text-gray-500 hover:text-purple-600 transition-colors" do %>
<%= @order.event.name %>
<% end %>
<svg class="w-4 h-4 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5l7 7-7 7"/>
</svg>
<li class="font-medium text-gray-900" aria-current="page">Commande #<%= @order.id %></li>
</ol>
</nav>
@@ -118,19 +113,9 @@
<!-- Order Total -->
<div class="border-t border-gray-200 pt-6">
<div class="space-y-2">
<div class="flex items-center justify-between">
<span class="text-gray-600">Sous-total</span>
<span class="text-gray-900"><%= @order.total_amount_euros - 1.0 %>€</span>
</div>
<div class="flex items-center justify-between">
<span class="text-gray-600">Frais de service</span>
<span class="text-gray-900">1.00€</span>
</div>
<div class="flex items-center justify-between text-lg pt-2 border-t border-gray-200">
<span class="font-medium text-gray-900">Total</span>
<span class="font-bold text-2xl text-purple-600"><%= @order.total_amount_euros %>€</span>
</div>
<div class="flex items-center justify-between text-lg">
<span class="font-medium text-gray-900">Total</span>
<span class="font-bold text-2xl text-purple-600"><%= @order.total_amount_euros %>€</span>
</div>
<p class="text-xs text-gray-500 mt-2">TVA incluse</p>
</div>

View File

@@ -1,131 +0,0 @@
<div class="container mx-auto px-4 sm:px-6 lg:px-8 py-8">
<!-- Breadcrumb -->
<nav class="flex my-6" aria-label="Breadcrumb">
<ol class="inline-flex items-center space-x-1 md:space-x-2 rounded-lg bg-white px-4 py-2 shadow-sm">
<li class="inline-flex items-center">
<%= link_to "Accueil", root_path, class: "inline-flex items-center text-sm font-medium text-gray-700 hover:text-purple-600" %>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<%= link_to "Tableau de bord", dashboard_path, class: "ml-1 text-sm font-medium text-gray-700 hover:text-purple-600 md:ml-2" %>
</div>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<span class="ml-1 text-sm font-medium text-purple-600 md:ml-2">Toutes mes commandes</span>
</div>
</li>
</ol>
</nav>
<!-- Header -->
<div class="flex items-center justify-between mb-8">
<div>
<h1 class="text-3xl font-bold text-slate-900 dark:text-slate-100">Toutes mes commandes</h1>
<p class="text-slate-600 dark:text-slate-400 mt-2">Consultez l'historique de toutes vos commandes</p>
</div>
<%= link_to dashboard_path, class: "inline-flex items-center px-4 py-2 bg-purple-100 hover:bg-purple-200 text-purple-700 font-medium rounded-lg transition-colors duration-200" do %>
<i data-lucide="arrow-left" class="w-4 h-4 mr-2"></i>
Retour au tableau de bord
<% end %>
</div>
<!-- Orders List -->
<% if @orders.any? %>
<div class="space-y-6">
<% @orders.each do |order| %>
<div class="card hover-lift">
<div class="card-body">
<div class="flex items-start justify-between mb-4">
<div class="flex-1">
<div class="flex items-center space-x-3 mb-2">
<h3 class="font-semibold text-slate-900 dark:text-slate-100"><%= order.event.name %></h3>
<span class="text-xs px-2 py-1 rounded-full <%= order.status == 'paid' ? 'bg-green-100 text-green-800 dark:bg-green-800 dark:text-green-100' : order.status == 'completed' ? 'bg-blue-100 text-blue-800 dark:bg-blue-800 dark:text-blue-100' : 'bg-yellow-100 text-yellow-800 dark:bg-yellow-800 dark:text-yellow-100' %>">
<%= order.status.humanize %>
</span>
</div>
<div class="flex items-center space-x-4 text-sm text-slate-600 dark:text-slate-400 mb-3">
<div class="flex items-center">
<i data-lucide="calendar" class="w-4 h-4 mr-1"></i>
<%= order.event.start_time.strftime("%d %B %Y à %H:%M") %>
</div>
<div class="flex items-center">
<i data-lucide="map-pin" class="w-4 h-4 mr-1"></i>
<%= order.event.venue_name %>
</div>
<div class="flex items-center">
<i data-lucide="shopping-bag" class="w-4 h-4 mr-1"></i>
<%= pluralize(order.tickets.count, 'billet') %>
</div>
</div>
<div class="text-sm text-slate-500 dark:text-slate-400">
Commande #<%= order.id %> • <%= order.created_at.strftime("%d/%m/%Y") %> • <%= order.total_amount_euros %>€
</div>
</div>
<div class="flex items-center space-x-2 ml-4">
<%= link_to order_path(order),
class: "inline-flex items-center px-3 py-2 bg-purple-600 hover:bg-purple-700 text-white text-sm font-medium rounded-lg transition-colors duration-200" do %>
<i data-lucide="eye" class="w-4 h-4 mr-2"></i>
Voir détails
<% end %>
</div>
</div>
<!-- Quick tickets preview -->
<div class="border-t border-slate-200 dark:border-slate-600 pt-3">
<div class="grid gap-2">
<% order.tickets.limit(3).each do |ticket| %>
<div class="flex items-center justify-between text-sm bg-slate-50 dark:bg-slate-700 rounded p-2">
<div class="flex items-center space-x-2">
<span class="w-2 h-2 bg-green-500 rounded-full"></span>
<span class="font-medium"><%= ticket.ticket_type.name %></span>
<span class="text-slate-500">- <%= ticket.first_name %> <%= ticket.last_name %></span>
</div>
<div class="flex items-center space-x-2">
<%= link_to ticket_download_path(ticket.qr_code),
class: "text-purple-600 hover:text-purple-800 dark:text-purple-400 dark:hover:text-purple-200" do %>
<i data-lucide="download" class="w-3 h-3"></i>
<% end %>
</div>
</div>
<% end %>
<% if order.tickets.count > 3 %>
<div class="text-xs text-slate-500 text-center">
et <%= order.tickets.count - 3 %> autre<%= order.tickets.count - 3 > 1 ? 's' : '' %> billet<%= order.tickets.count - 3 > 1 ? 's' : '' %>
</div>
<% end %>
</div>
</div>
</div>
</div>
<% end %>
</div>
<!-- Pagination -->
<div class="mt-8">
<%= paginate @orders %>
</div>
<% else %>
<div class="text-center py-12">
<div class="w-16 h-16 bg-slate-100 dark:bg-slate-700 rounded-full flex items-center justify-center mx-auto mb-4">
<i data-lucide="shopping-bag" class="w-8 h-8 text-slate-400"></i>
</div>
<h3 class="text-lg font-medium text-slate-900 dark:text-slate-100 mb-2">Aucune commande</h3>
<p class="text-slate-600 dark:text-slate-400 mb-6">Vous n'avez encore passé aucune commande.</p>
<%= link_to events_path, class: "inline-flex items-center px-4 py-2 bg-purple-600 text-white rounded-lg hover:bg-purple-700 transition-colors duration-200" do %>
<i data-lucide="search" class="w-4 h-4 mr-2"></i>
Découvrir les événements
<% end %>
</div>
<% end %>
</div>

View File

@@ -1,173 +0,0 @@
<div class="min-h-screen bg-gradient-to-br from-purple-50 to-indigo-50 py-8">
<div class="max-w-4xl mx-auto px-4 sm:px-6 lg:px-8">
<!-- Breadcrumb -->
<nav class="flex mb-6" aria-label="Breadcrumb">
<ol class="inline-flex items-center space-x-1 md:space-x-2 rounded-lg bg-white px-4 py-2 shadow-sm">
<li class="inline-flex items-center">
<%= link_to "Accueil", root_path, class: "inline-flex items-center text-sm font-medium text-gray-700 hover:text-purple-600" %>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<%= link_to "Tableau de bord", dashboard_path, class: "ml-1 text-sm font-medium text-gray-700 hover:text-purple-600 md:ml-2" %>
</div>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<%= link_to "Commande ##{@order.id}", order_path(@order), class: "ml-1 text-sm font-medium text-gray-700 hover:text-purple-600 md:ml-2" %>
</div>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<span class="ml-1 text-sm font-medium text-purple-600 md:ml-2">Facture</span>
</div>
</li>
</ol>
</nav>
<!-- Invoice Header -->
<div class="bg-white rounded-2xl shadow-xl p-6 md:p-8 mb-8">
<div class="flex flex-col md:flex-row md:items-center md:justify-between mb-8">
<div>
<h1 class="text-3xl font-bold text-gray-900 mb-2">Facture</h1>
<p class="text-gray-600">Commande #<%= @order.id %> • <%= @order.created_at.strftime("%d %B %Y") %></p>
</div>
<div class="mt-4 md:mt-0">
<% if @stripe_invoice_pdf_url %>
<%= link_to @stripe_invoice_pdf_url, target: "_blank", class: "inline-flex items-center px-4 py-2 bg-purple-600 text-white font-medium rounded-lg hover:bg-purple-700 transition-colors" do %>
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z" />
</svg>
Télécharger la facture (PDF)
<% end %>
<% end %>
</div>
</div>
<!-- Invoice Details -->
<div class="grid grid-cols-1 md:grid-cols-2 gap-8 mb-8">
<!-- From -->
<div>
<h3 class="text-lg font-semibold text-gray-900 mb-3">Émis par</h3>
<div class="bg-purple-50 rounded-lg p-4 border border-purple-200">
<h4 class="font-semibold text-purple-900">AperoNight</h4>
<div class="mt-2 space-y-1 text-sm text-purple-700">
<p>123 Avenue des Événements</p>
<p>75000 Paris, France</p>
<p>contact@apero-night.fr</p>
</div>
</div>
</div>
<!-- To -->
<div>
<h3 class="text-lg font-semibold text-gray-900 mb-3">Facturé à</h3>
<div class="bg-gray-50 rounded-lg p-4 border border-gray-200">
<h4 class="font-semibold text-gray-900">
<%= @order.user.first_name %> <%= @order.user.last_name %>
</h4>
<div class="mt-2 space-y-1 text-sm text-gray-600">
<p><%= @order.user.email %></p>
<% if @order.user.company_name.present? %>
<p><%= @order.user.company_name %></p>
<% end %>
</div>
</div>
</div>
</div>
<!-- Event Information -->
<div class="mb-8">
<h3 class="text-lg font-semibold text-gray-900 mb-3">Événement</h3>
<div class="bg-indigo-50 rounded-lg p-4 border border-indigo-200">
<h4 class="font-semibold text-indigo-900 text-lg"><%= @order.event.name %></h4>
<div class="mt-2 space-y-1 text-sm text-indigo-700">
<% if @order.event.start_time %>
<div class="flex items-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z" />
</svg>
<%= @order.event.start_time.strftime("%d %B %Y à %H:%M") %>
</div>
<% end %>
<% if @order.event.venue_name.present? %>
<div class="flex items-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M17.657 16.657L13.414 20.9a1.998 1.998 0 01-2.827 0l-4.244-4.243a8 8 0 1111.314 0z" />
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 11a3 3 0 11-6 0 3 3 0 016 0z" />
</svg>
<%= @order.event.venue_name %>
</div>
<% end %>
</div>
</div>
</div>
<!-- Items -->
<div class="mb-8">
<h3 class="text-lg font-semibold text-gray-900 mb-4">Détails de la facture</h3>
<div class="overflow-hidden border border-gray-200 rounded-lg">
<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">Description</th>
<th scope="col" class="px-6 py-3 text-right text-xs font-medium text-gray-500 uppercase tracking-wider">Quantité</th>
<th scope="col" class="px-6 py-3 text-right text-xs font-medium text-gray-500 uppercase tracking-wider">Prix unitaire</th>
<th scope="col" class="px-6 py-3 text-right text-xs font-medium text-gray-500 uppercase tracking-wider">Total</th>
</tr>
</thead>
<tbody class="bg-white divide-y divide-gray-200">
<% @tickets.group_by(&:ticket_type).each do |ticket_type, tickets| %>
<tr>
<td class="px-6 py-4 whitespace-nowrap">
<div class="text-sm font-medium text-gray-900"><%= ticket_type.name %></div>
<div class="text-sm text-gray-500"><%= ticket_type.description %></div>
</td>
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500 text-right"><%= tickets.count %></td>
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500 text-right"><%= "%.2f" % (ticket_type.price_cents / 100.0) %>€</td>
<td class="px-6 py-4 whitespace-nowrap text-sm font-medium text-gray-900 text-right"><%= "%.2f" % (tickets.count * ticket_type.price_cents / 100.0) %>€</td>
</tr>
<% end %>
<tr>
<td class="px-6 py-4 whitespace-nowrap text-sm font-medium text-gray-900">Frais de service</td>
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500 text-right">1</td>
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500 text-right">1.00€</td>
<td class="px-6 py-4 whitespace-nowrap text-sm font-medium text-gray-900 text-right">1.00€</td>
</tr>
</tbody>
<tfoot class="bg-gray-50">
<tr>
<th colspan="3" scope="col" class="px-6 py-3 text-right text-sm font-medium text-gray-900 uppercase tracking-wider">Total</th>
<th scope="col" class="px-6 py-3 text-right text-lg font-bold text-gray-900"><%= "%.2f" % @order.total_amount_euros %>€</th>
</tr>
</tfoot>
</table>
</div>
</div>
<!-- Payment Information -->
<div class="border-t border-gray-200 pt-6">
<h3 class="text-lg font-semibold text-gray-900 mb-3">Paiement</h3>
<div class="flex items-center">
<div class="flex-shrink-0 w-8 h-8 bg-green-100 rounded-full flex items-center justify-center">
<svg class="w-4 h-4 text-green-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M5 13l4 4L19 7" />
</svg>
</div>
<div class="ml-4">
<h4 class="font-medium text-gray-900">Paiement effectué</h4>
<p class="text-sm text-gray-600">Commande #<%= @order.id %> payée le <%= @order.updated_at.strftime("%d %B %Y") %></p>
</div>
</div>
</div>
</div>
</div>
</div>

View File

@@ -1,34 +1,31 @@
<div class="min-h-screen bg-gradient-to-br from-gray-50 to-gray-100">
<div class="max-w-4xl mx-auto px-4 sm:px-6 lg:px-8 py-8">
<!-- Breadcrumb -->
<nav class="flex mb-6" aria-label="Breadcrumb">
<ol class="inline-flex items-center space-x-1 md:space-x-2 rounded-lg bg-white px-4 py-2 shadow-sm">
<li class="inline-flex items-center">
<%= link_to "Accueil", root_path, class: "inline-flex items-center text-sm font-medium text-gray-700 hover:text-purple-600" %>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<%= link_to "Événements", events_path, class: "ml-1 text-sm font-medium text-gray-700 hover:text-purple-600 md:ml-2" %>
</div>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<%= link_to @event.name, event_path(@event.slug, @event), class: "ml-1 text-sm font-medium text-gray-700 hover:text-purple-600 md:ml-2" %>
</div>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<span class="ml-1 text-sm font-medium text-purple-600 md:ml-2">Nouvelle commande</span>
</div>
<nav class="mb-8" aria-label="Breadcrumb">
<ol class="flex items-center space-x-2 text-sm">
<%= link_to root_path, class: "text-gray-500 hover:text-purple-600 transition-colors" do %>
<svg class="w-4 h-4 inline-block mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M3 7v10a2 2 0 002 2h14a2 2 0 002-2V9a2 2 0 00-2-2h-6l-2-2H5a2 2 0 00-2 2z"/>
</svg>
Accueil
<% end %>
<svg class="w-4 h-4 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5l7 7-7 7"/>
</svg>
<%= link_to events_path, class: "text-gray-500 hover:text-purple-600 transition-colors" do %>
Événements
<% end %>
<svg class="w-4 h-4 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5l7 7-7 7"/>
</svg>
<%= link_to event_path(@event.slug, @event), class: "text-gray-500 hover:text-purple-600 transition-colors" do %>
<%= @event.name %>
<% end %>
<svg class="w-4 h-4 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5l7 7-7 7"/>
</svg>
<li class="font-medium text-gray-900" aria-current="page">
Nouvelle commande
</li>
</ol>
</nav>

View File

@@ -1,198 +0,0 @@
<div class="min-h-screen bg-gradient-to-br from-purple-50 to-indigo-50 py-8">
<div class="max-w-4xl mx-auto px-4 sm:px-6 lg:px-8">
<!-- Breadcrumb -->
<nav class="flex mb-6" aria-label="Breadcrumb">
<ol class="inline-flex items-center space-x-1 md:space-x-2 rounded-lg bg-white px-4 py-2 shadow-sm">
<li class="inline-flex items-center">
<%= link_to "Accueil", root_path, class: "inline-flex items-center text-sm font-medium text-gray-700 hover:text-purple-600" %>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<%= link_to "Tableau de bord", dashboard_path, class: "ml-1 text-sm font-medium text-gray-700 hover:text-purple-600 md:ml-2" %>
</div>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<span class="ml-1 text-sm font-medium text-purple-600 md:ml-2">Commande #<%= @order&.id || 'Inconnue' %></span>
</div>
</li>
</ol>
</nav>
<!-- Header -->
<div class="text-center mb-8">
<div class="mx-auto w-16 h-16 bg-purple-100 rounded-full flex items-center justify-center mb-4">
<svg class="w-8 h-8 text-purple-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
</div>
<h1 class="text-3xl font-bold text-gray-900 mb-2">Détails de la Commande</h1>
</div>
<div class="grid grid-cols-1 lg:grid-cols-2 gap-8">
<!-- Event & Order Details -->
<div class="bg-white rounded-2xl shadow-xl p-6 md:p-8">
<div class="border-b border-gray-200 pb-6 mb-6">
<h2 class="text-2xl font-bold text-gray-900 mb-2">Détails de Votre Commande</h2>
<% if @order %>
<div class="flex items-center text-sm text-gray-600 space-x-4">
<div class="flex items-center">
<svg class="w-4 h-4 mr-2 flex-shrink-0" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
<div class="flex flex-col">
<span class="font-medium">Commande n°<%= @order.id %></span>
<span class="text-xs text-gray-500"><%= @order.created_at.strftime("%d %B %Y") %></span>
</div>
</div>
<div class="flex items-center">
<svg class="w-4 h-4 mr-1 text-red-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4m0 4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
<span class="text-red-600 font-medium">
Paiement annulé
</span>
</div>
</div>
<% else %>
<p class="text-gray-600">Aucune commande trouvée.</p>
<% end %>
</div>
<% if @order %>
<!-- Event Information -->
<div class="mb-6">
<h3 class="text-lg font-semibold text-gray-900 mb-3">Événement</h3>
<div class="bg-purple-50 rounded-lg p-4 border border-purple-200">
<h4 class="font-semibold text-purple-900 text-lg"><%= @order.event.name %></h4>
<div class="mt-2 space-y-1 text-sm text-purple-700">
<% if @order.event.start_time %>
<div class="flex items-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
<%= @order.event.start_time.strftime("%d %B %Y à %H:%M") %>
</div>
<% end %>
<% if @order.event.venue_name.present? %>
<div class="flex items-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M17.657 16.657L13.414 20.9a1.998 1.998 0 01-2.827 0l-4.244-4.243a8 8 0 1111.314 0z"/>
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 11a3 3 0 11-6 0 3 3 0 016 0z"/>
</svg>
<%= @order.event.venue_name %>
</div>
<% end %>
<% if @order.event.venue_address.present? %>
<div class="flex items-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 20l-5.447-2.724A1 1 0 013 16.382V5.618a1 1 0 011.447-.894L9 7m0 13l6-3m-6 3V7m6 10l4.553 2.276A1 1 0 0021 18.382V7.618a1 1 0 00-.553-.894L15 4m0 13V4m0 0L9 7"/>
</svg>
<%= @order.event.venue_address %>
</div>
<% end %>
</div>
</div>
</div>
<!-- Summary -->
<div class="space-y-4">
<h3 class="text-lg font-semibold text-gray-900 mb-4">Récapitulatif</h3>
<% @order.tickets.each do |ticket| %>
<div class="flex items-center justify-between py-3 border-b border-gray-100 last:border-b-0">
<div class="flex-1 min-w-0">
<h4 class="text-sm font-medium text-gray-900 truncate"><%= ticket.ticket_type.name %></h4>
<div class="flex items-center text-xs text-gray-500 mt-1">
<svg class="w-3 h-3 mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M16 7a4 4 0 11-8 0 4 4 0 018 0zM12 14a7 7 0 00-7 7h14a7 7 0 00-7-7z"/>
</svg>
<%= ticket.first_name %> <%= ticket.last_name %>
</div>
<div class="flex items-center text-xs text-red-600 mt-1">
<svg class="w-3 h-3 mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4m0 4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
En attente de paiement
</div>
</div>
<div class="text-right">
<div class="text-lg font-semibold text-gray-900"><%= ticket.price_euros %>€</div>
</div>
</div>
<% end %>
</div>
<!-- Total -->
<div class="border-t border-gray-200 pt-6 mt-6">
<div class="space-y-2">
<div class="flex items-center justify-between">
<span class="text-gray-600">Sous-total</span>
<span class="text-gray-900"><%= @order.total_amount_euros - 1.0 %>€</span>
</div>
<div class="flex items-center justify-between">
<span class="text-gray-600">Frais de service</span>
<span class="text-gray-900">1.00€</span>
</div>
<div class="flex items-center justify-between text-lg pt-2 border-t border-gray-200">
<span class="font-medium text-gray-900">Total à payer</span>
<span class="font-bold text-2xl text-red-600">
<%= @order.total_amount_euros %>€
</span>
</div>
</div>
</div>
<% end %>
</div>
<!-- Actions & Ticket Access -->
<div class="bg-white rounded-2xl shadow-xl p-6 md:p-8 h-fit">
<% if @order&.can_retry_payment? %>
<!-- Payment Required -->
<div class="border-b border-gray-200 pb-6 mb-6">
<h2 class="text-xl font-bold text-gray-900 mb-2">Paiement Requis</h2>
<p class="text-sm text-gray-600">Votre commande nécessite un paiement</p>
</div>
<div class="mb-6">
<%= link_to checkout_order_path(@order), class: "block w-full text-center py-3 px-4 bg-orange-600 hover:bg-orange-700 text-white font-medium rounded-lg transition-colors" do %>
<div class="flex items-center justify-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M3 10h18M7 15h1m4 0h1m-7 4h12a3 3 0 003-3V8a3 3 0 00-3-3H6a3 3 0 00-3 3v8a3 3 0 003 3z"/>
</svg>
Procéder au Paiement
</div>
<% end %>
</div>
<% end %>
<!-- Navigation Actions -->
<div class="border-t border-gray-200 pt-6 mt-6">
<div class="space-y-3">
<%= link_to dashboard_path, class: "block w-full text-center py-3 px-4 bg-purple-600 hover:bg-purple-700 text-white font-medium rounded-lg transition-colors" do %>
<div class="flex items-center justify-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10 19l-7-7m0 0l7-7m-7 7h18"/>
</svg>
Retour au Tableau de Bord
</div>
<% end %>
<%= link_to event_path(@order.event.slug, @order.event), class: "block w-full text-center py-3 px-4 border border-gray-300 rounded-lg text-gray-700 hover:bg-gray-50 transition-colors" do %>
<div class="flex items-center justify-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z"/>
</svg>
Voir l'Événement Complet
</div>
<% end %>
</div>
</div>
</div>
</div>
</div>
</div>

View File

@@ -1,61 +1,35 @@
<div class="min-h-screen bg-gradient-to-br from-purple-50 to-indigo-50 py-8">
<div class="min-h-screen bg-gradient-to-br from-green-50 to-emerald-50 py-8">
<div class="max-w-4xl mx-auto px-4 sm:px-6 lg:px-8">
<!-- Breadcrumb -->
<nav class="flex mb-6" aria-label="Breadcrumb">
<ol class="inline-flex items-center space-x-1 md:space-x-2 rounded-lg bg-white px-4 py-2 shadow-sm">
<li class="inline-flex items-center">
<%= link_to "Accueil", root_path, class: "inline-flex items-center text-sm font-medium text-gray-700 hover:text-purple-600" %>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<%= link_to "Tableau de bord", dashboard_path, class: "ml-1 text-sm font-medium text-gray-700 hover:text-purple-600 md:ml-2" %>
</div>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<span class="ml-1 text-sm font-medium text-purple-600 md:ml-2">Commande #<%= @order.id %></span>
</div>
</li>
</ol>
</nav>
<!-- Header -->
<div class="text-center mb-8">
<div class="mx-auto w-16 h-16 bg-purple-100 rounded-full flex items-center justify-center mb-4">
<svg class="w-8 h-8 text-purple-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
<!-- Success Header -->
<div class="text-center mb-12">
<div class="mx-auto w-20 h-20 bg-green-100 rounded-full flex items-center justify-center mb-6">
<svg class="w-10 h-10 text-green-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M5 13l4 4L19 7"/>
</svg>
</div>
<h1 class="text-3xl font-bold text-gray-900 mb-2">Détails de la Commande</h1>
<h1 class="text-4xl font-bold text-gray-900 mb-4">Paiement réussi !</h1>
<p class="text-xl text-gray-600 max-w-2xl mx-auto">
Félicitations ! Votre commande a été traitée avec succès. Vous allez recevoir vos billets par email d'ici quelques minutes.
</p>
</div>
<div class="grid grid-cols-1 lg:grid-cols-2 gap-8">
<!-- Event & Order Details -->
<!-- Order Summary -->
<div class="bg-white rounded-2xl shadow-xl p-6 md:p-8">
<div class="border-b border-gray-200 pb-6 mb-6">
<h2 class="text-2xl font-bold text-gray-900 mb-2">Détails de Votre Commande</h2>
<h2 class="text-2xl font-bold text-gray-900 mb-2">Récapitulatif de la commande</h2>
<div class="flex items-center text-sm text-gray-600 space-x-4">
<div class="flex items-center">
<svg class="w-4 h-4 mr-2 flex-shrink-0" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<svg class="w-4 h-4 mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
<div class="flex flex-col">
<span class="font-medium">Commande n°<%= @order.id %></span>
<span class="text-xs text-gray-500"><%= @order.created_at.strftime("%d %B %Y") %></span>
</div>
Commande #<%= @order.id %>
</div>
<div class="flex items-center">
<svg class="w-4 h-4 mr-1 text-green-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M5 13l4 4L19 7"/>
</svg>
<span class="text-green-600 font-medium">
Payée
</span>
<span class="text-green-600 font-medium">Payée</span>
</div>
</div>
</div>
@@ -71,7 +45,7 @@
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
<%= @order.event.start_time.strftime("%d %B %Y à %H:%M") %>
<%= l(@order.event.start_time, format: :long) %>
</div>
<% end %>
<% if @order.event.venue_name.present? %>
@@ -83,21 +57,13 @@
<%= @order.event.venue_name %>
</div>
<% end %>
<% if @order.event.venue_address.present? %>
<div class="flex items-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 20l-5.447-2.724A1 1 0 013 16.382V5.618a1 1 0 011.447-.894L9 7m0 13l6-3m-6 3V7m6 10l4.553 2.276A1 1 0 0021 18.382V7.618a1 1 0 00-.553-.894L15 4m0 13V4m0 0L9 7"/>
</svg>
<%= @order.event.venue_address %>
</div>
<% end %>
</div>
</div>
</div>
<!-- Summary -->
<!-- Tickets List -->
<div class="space-y-4">
<h3 class="text-lg font-semibold text-gray-900 mb-4">Récapitulatif</h3>
<h3 class="text-lg font-semibold text-gray-900 mb-4">Vos billets</h3>
<% @order.tickets.each do |ticket| %>
<div class="flex items-center justify-between py-3 border-b border-gray-100 last:border-b-0">
@@ -125,59 +91,48 @@
<!-- Total -->
<div class="border-t border-gray-200 pt-6 mt-6">
<div class="space-y-2">
<div class="flex items-center justify-between">
<span class="text-gray-600">Sous-total</span>
<span class="text-gray-900"><%= @order.total_amount_euros - 1.0 %>€</span>
</div>
<div class="flex items-center justify-between">
<span class="text-gray-600">Frais de service</span>
<span class="text-gray-900">1.00€</span>
</div>
<div class="flex items-center justify-between text-lg pt-2 border-t border-gray-200">
<span class="font-medium text-gray-900">Total payé</span>
<span class="font-bold text-2xl text-green-600">
<%= @order.total_amount_euros %>€
</span>
</div>
<div class="flex items-center justify-between text-lg">
<span class="font-medium text-gray-900">Total payé</span>
<span class="font-bold text-2xl text-green-600"><%= @order.total_amount_euros %>€</span>
</div>
</div>
</div>
<!-- Actions & Ticket Access -->
<!-- Next Steps -->
<div class="bg-white rounded-2xl shadow-xl p-6 md:p-8 h-fit">
<!-- Ticket Access -->
<div class="border-b border-gray-200 pb-6 mb-6">
<h2 class="text-xl font-bold text-gray-900 mb-2">Accédez à Vos Billets</h2>
<p class="text-sm text-gray-600">Téléchargez ou consultez vos billets</p>
<h2 class="text-xl font-bold text-gray-900 mb-2">Prochaines étapes</h2>
<p class="text-sm text-gray-600">Que faire maintenant ?</p>
</div>
<div class="space-y-6">
<!-- Email Confirmation -->
<div class="flex items-start">
<div class="flex-shrink-0 w-8 h-8 bg-blue-100 rounded-full flex items-center justify-center">
<span class="text-blue-600 font-semibold text-sm">1</span>
</div>
<div class="ml-4">
<h3 class="font-semibold text-gray-900 mb-1">Vérifiez votre email</h3>
<p class="text-gray-600 text-sm">Nous avons envoyé vos billets à <strong><%= current_user.email %></strong>. Vérifiez aussi vos spams.</p>
</div>
</div>
<!-- Download Tickets -->
<div class="flex items-start">
<div class="flex-shrink-0 w-8 h-8 bg-purple-100 rounded-full flex items-center justify-center">
<svg class="w-4 h-4 text-purple-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
<span class="text-purple-600 font-semibold text-sm">2</span>
</div>
<div class="ml-4">
<h3 class="font-semibold text-gray-900 mb-1">Télécharger Vos Billets</h3>
<h3 class="font-semibold text-gray-900 mb-1">Téléchargez vos billets</h3>
<p class="text-gray-600 text-sm mb-3">Gardez vos billets sur votre téléphone ou imprimez-les.</p>
<div class="space-y-2">
<% @order.tickets.each_with_index do |ticket, index| %>
<div class="flex items-center justify-between p-3 border border-purple-200 rounded-lg bg-purple-50 hover:bg-purple-100 transition-colors">
<%= link_to ticket_path(ticket.qr_code), class: "flex-1 flex items-center text-purple-700 hover:text-purple-800 font-medium" do %>
<div class="flex items-center justify-center w-6 h-6 bg-purple-200 text-purple-800 text-xs font-bold rounded-full mr-3">
<%= index + 1 %>
</div>
<%= ticket.first_name %> <%= ticket.last_name %>
<% end %>
<%= link_to ticket_download_path(ticket.qr_code), class: "ml-3 p-2 text-purple-600 hover:text-purple-800 hover:bg-purple-200 rounded-lg transition-colors", title: "Télécharger le billet PDF" do %>
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
<% end %>
</div>
<% @order.tickets.each do |ticket| %>
<%= link_to download_ticket_path(ticket), class: "inline-flex items-center px-3 py-2 border border-purple-300 rounded-md text-sm font-medium text-purple-700 bg-purple-50 hover:bg-purple-100 transition-colors mr-2 mb-2" do %>
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
<%= ticket.first_name %> <%= ticket.last_name %>
<% end %>
<% end %>
</div>
</div>
@@ -186,34 +141,46 @@
<!-- Event Day -->
<div class="flex items-start">
<div class="flex-shrink-0 w-8 h-8 bg-green-100 rounded-full flex items-center justify-center">
<svg class="w-4 h-4 text-green-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
<span class="text-green-600 font-semibold text-sm">3</span>
</div>
<div class="ml-4">
<h3 class="font-semibold text-gray-900 mb-1">Le Jour de l'Événement</h3>
<h3 class="font-semibold text-gray-900 mb-1">Le jour J</h3>
<p class="text-gray-600 text-sm">Présentez votre billet (QR code) à l'entrée. Arrivez un peu en avance !</p>
</div>
</div>
</div>
<!-- Navigation Actions -->
<!-- Contact Support -->
<div class="bg-gray-50 rounded-lg p-4 mt-8">
<h4 class="font-medium text-gray-900 mb-2">Besoin d'aide ?</h4>
<p class="text-gray-600 text-sm mb-3">Si vous avez des questions ou des problèmes avec votre commande, n'hésitez pas à nous contacter.</p>
<div class="space-y-2">
<%= link_to "mailto:support@example.com", class: "inline-flex items-center text-sm text-purple-600 hover:text-purple-700" do %>
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M3 8l7.89 4.26a2 2 0 002.22 0L21 8M5 19h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v10a2 2 0 002 2z"/>
</svg>
Contactez le support
<% end %>
</div>
</div>
<!-- Actions -->
<div class="border-t border-gray-200 pt-6 mt-6">
<div class="space-y-3">
<%= link_to dashboard_path, class: "block w-full text-center py-3 px-4 bg-purple-600 hover:bg-purple-700 text-white font-medium rounded-lg transition-colors" do %>
<div class="flex items-center justify-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10 19l-7-7m0 0l7-7m-7 7h18"/>
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M3 7v10a2 2 0 002 2h14a2 2 0 002-2V9a2 2 0 00-2-2h-6l-2-2H5a2 2 0 00-2 2z"/>
</svg>
Retour au Tableau de Bord
Voir tous mes billets
</div>
<% end %>
<%= link_to event_path(@order.event.slug, @order.event), class: "block w-full text-center py-3 px-4 border border-gray-300 rounded-lg text-gray-700 hover:bg-gray-50 transition-colors" do %>
<%= link_to events_path, class: "block w-full text-center py-3 px-4 border border-gray-300 rounded-lg text-gray-700 hover:bg-gray-50 transition-colors" do %>
<div class="flex items-center justify-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z"/>
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z"/>
</svg>
Voir l'Événement Complet
Découvrir d'autres événements
</div>
<% end %>
</div>

View File

@@ -1,278 +1,102 @@
<div class="min-h-screen bg-gradient-to-br from-purple-50 to-indigo-50 py-8">
<div class="min-h-screen bg-gradient-to-br from-gray-50 to-gray-100 py-8">
<div class="max-w-4xl mx-auto px-4 sm:px-6 lg:px-8">
<!-- Breadcrumb -->
<nav class="flex mb-6" aria-label="Breadcrumb">
<ol class="inline-flex items-center space-x-1 md:space-x-2 rounded-lg bg-white px-4 py-2 shadow-sm">
<li class="inline-flex items-center">
<%= link_to "Accueil", root_path, class: "inline-flex items-center text-sm font-medium text-gray-700 hover:text-purple-600" %>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<%= link_to "Tableau de bord", dashboard_path, class: "ml-1 text-sm font-medium text-gray-700 hover:text-purple-600 md:ml-2" %>
</div>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<span class="ml-1 text-sm font-medium text-purple-600 md:ml-2">Commande #<%= @order.id %></span>
</div>
</li>
<nav class="mb-8" aria-label="Breadcrumb">
<ol class="flex items-center space-x-2 text-sm">
<%= link_to root_path, class: "text-gray-500 hover:text-purple-600 transition-colors" do %>
<svg class="w-4 h-4 inline-block mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M3 7v10a2 2 0 002 2h14a2 2 0 002-2V9a2 2 0 00-2-2h-6l-2-2H5a2 2 0 00-2 2z"/>
</svg>
Accueil
<% end %>
<svg class="w-4 h-4 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5l7 7-7 7"/>
</svg>
<%= link_to events_path, class: "text-gray-500 hover:text-purple-600 transition-colors" do %>
Événements
<% end %>
<svg class="w-4 h-4 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5l7 7-7 7"/>
</svg>
<%= link_to event_path(@order.event.slug, @order.event), class: "text-gray-500 hover:text-purple-600 transition-colors" do %>
<%= @order.event.name %>
<% end %>
<svg class="w-4 h-4 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5l7 7-7 7"/>
</svg>
<li class="font-medium text-gray-900" aria-current="page">Commande #<%= @order.id %></li>
</ol>
</nav>
<!-- Header -->
<div class="text-center mb-8">
<div class="mx-auto w-16 h-16 bg-purple-100 rounded-full flex items-center justify-center mb-4">
<svg class="w-8 h-8 text-purple-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
<div class="bg-white rounded-2xl shadow-xl p-6 md:p-8">
<div class="border-b border-gray-200 pb-6 mb-6">
<h1 class="text-2xl font-bold text-gray-900 mb-2">Détails de la commande</h1>
<div class="flex items-center text-sm text-gray-600 space-x-4">
<div class="flex items-center">
<svg class="w-4 h-4 mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
Commande #<%= @order.id %>
</div>
<div class="flex items-center">
<svg class="w-4 h-4 mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
<%= @order.status.titleize %>
</div>
</div>
</div>
<h1 class="text-3xl font-bold text-gray-900 mb-2">Détails de la Commande</h1>
</div>
<div class="grid grid-cols-1 lg:grid-cols-2 gap-8">
<!-- Event & Order Details -->
<div class="bg-white rounded-2xl shadow-xl p-6 md:p-8">
<div class="border-b border-gray-200 pb-6 mb-6">
<h2 class="text-2xl font-bold text-gray-900 mb-2">Informations</h2>
<div class="flex items-center text-sm text-gray-600 space-x-4">
<div class="flex items-center">
<svg class="w-4 h-4 mr-2 flex-shrink-0" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
<div class="flex flex-col">
<span class="font-medium">Commande n°<%= @order.id %></span>
<span class="text-xs text-gray-500"><%= @order.created_at.strftime("%d %B %Y") %></span>
</div>
</div>
<div class="flex items-center">
<svg class="w-4 h-4 mr-1 <%= @order.status == 'paid' || @order.status == 'completed' ? 'text-green-600' : 'text-yellow-600' %>" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<% if @order.status == 'paid' || @order.status == 'completed' %>
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M5 13l4 4L19 7"/>
<% else %>
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"/>
<% end %>
</svg>
<span class="<%= @order.status == 'paid' || @order.status == 'completed' ? 'text-green-600' : 'text-yellow-600' %> font-medium">
<%= case @order.status
when 'paid' then 'Payé'
when 'completed' then 'Terminé'
else @order.status.humanize
end %>
</span>
</div>
</div>
</div>
<!-- Event Information -->
<div class="mb-6">
<h3 class="text-lg font-semibold text-gray-900 mb-3">Événement</h3>
<div class="bg-purple-50 rounded-lg p-4 border border-purple-200">
<h4 class="font-semibold text-purple-900 text-lg"><%= @order.event.name %></h4>
<div class="mt-2 space-y-1 text-sm text-purple-700">
<% if @order.event.start_time %>
<div class="flex items-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
<%= @order.event.start_time.strftime("%d %B %Y à %H:%M") %>
</div>
<% end %>
<% if @order.event.venue_name.present? %>
<div class="flex items-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M17.657 16.657L13.414 20.9a1.998 1.998 0 01-2.827 0l-4.244-4.243a8 8 0 1111.314 0z"/>
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 11a3 3 0 11-6 0 3 3 0 016 0z"/>
</svg>
<%= @order.event.venue_name %>
</div>
<% end %>
<% if @order.event.venue_address.present? %>
<div class="flex items-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 20l-5.447-2.724A1 1 0 013 16.382V5.618a1 1 0 011.447-.894L9 7m0 13l6-3m-6 3V7m6 10l4.553 2.276A1 1 0 0021 18.382V7.618a1 1 0 00-.553-.894L15 4m0 13V4m0 0L9 7"/>
</svg>
<%= @order.event.venue_address %>
</div>
<% end %>
</div>
</div>
</div>
<!-- Summary -->
<div class="space-y-4">
<h3 class="text-lg font-semibold text-gray-900 mb-4">Récapitulatif</h3>
<% @tickets.each do |ticket| %>
<div class="flex items-center justify-between py-3 border-b border-gray-100 last:border-b-0">
<div class="flex-1 min-w-0">
<h4 class="text-sm font-medium text-gray-900 truncate"><%= ticket.ticket_type.name %></h4>
<div class="flex items-center text-xs text-gray-500 mt-1">
<svg class="w-3 h-3 mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M16 7a4 4 0 11-8 0 4 4 0 018 0zM12 14a7 7 0 00-7 7h14a7 7 0 00-7-7z"/>
</svg>
<%= ticket.first_name %> <%= ticket.last_name %>
</div>
<% if @order.status == 'paid' || @order.status == 'completed' %>
<div class="flex items-center text-xs text-green-600 mt-1">
<svg class="w-3 h-3 mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M5 13l4 4L19 7"/>
</svg>
Actif
</div>
<% end %>
</div>
<div class="text-right">
<div class="text-lg font-semibold text-gray-900"><%= ticket.price_euros %>€</div>
</div>
</div>
<% end %>
</div>
<!-- Total -->
<div class="border-t border-gray-200 pt-6 mt-6">
<div class="space-y-2">
<div class="flex items-center justify-between">
<span class="text-gray-600">Sous-total</span>
<span class="text-gray-900"><%= @order.total_amount_euros - 1.0 %>€</span>
</div>
<div class="flex items-center justify-between">
<span class="text-gray-600">Frais de service</span>
<span class="text-gray-900">1.00€</span>
</div>
<div class="flex items-center justify-between text-lg pt-2 border-t border-gray-200">
<span class="font-medium text-gray-900">Total <%= @order.status == 'paid' || @order.status == 'completed' ? 'payé' : 'à payer' %></span>
<span class="font-bold text-2xl <%= @order.status == 'paid' || @order.status == 'completed' ? 'text-green-600' : 'text-purple-600' %>">
<%= @order.total_amount_euros %>€
</span>
</div>
</div>
</div>
<!-- View Invoice -->
<% if @order.status == 'paid' || @order.status == 'completed' %>
<div class="border-t border-gray-200 pt-6 mt-6">
<div class="flex items-start">
<div class="flex-shrink-0 w-8 h-8 bg-blue-100 rounded-full flex items-center justify-center">
<svg class="w-4 h-4 text-blue-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
<!-- Order Items -->
<div class="space-y-4 mb-6">
<h3 class="text-lg font-semibold text-gray-900 mb-4">Billets commandés</h3>
<% @tickets.each do |ticket| %>
<div class="flex items-center justify-between py-4 border-b border-gray-100 last:border-b-0">
<div class="flex-1 min-w-0">
<h4 class="text-sm font-medium text-gray-900"><%= ticket.ticket_type.name %></h4>
<div class="flex items-center text-xs text-gray-500 mt-1">
<svg class="w-3 h-3 mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M16 7a4 4 0 11-8 0 4 4 0 018 0zM12 14a7 7 0 00-7 7h14a7 7 0 00-7-7z"/>
</svg>
<%= ticket.first_name %> <%= ticket.last_name %>
</div>
<div class="ml-4">
<h3 class="font-semibold text-gray-900 mb-1">Consulter la Facture</h3>
<p class="text-gray-600 text-sm mb-3">Téléchargez ou consultez la facture de votre commande.</p>
<div class="mt-2">
<%= link_to invoice_order_path(@order), class: "inline-flex items-center px-4 py-2 bg-blue-600 text-white font-medium rounded-lg hover:bg-blue-700 transition-colors" do %>
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
Voir la facture
<% end %>
</div>
<div class="text-xs text-gray-500 mt-1">
Statut: <%= ticket.status.titleize %>
</div>
</div>
<div class="text-right">
<div class="text-lg font-semibold text-gray-900"><%= ticket.price_euros %>€</div>
</div>
</div>
<% end %>
</div>
<!-- Actions & Ticket Access -->
<div class="bg-white rounded-2xl shadow-xl p-6 md:p-8 h-fit">
<% if @order.status == 'paid' || @order.status == 'completed' %>
<!-- Ticket Access -->
<div class="border-b border-gray-200 pb-6 mb-6">
<h2 class="text-xl font-bold text-gray-900 mb-2">Accédez à Vos Billets</h2>
<p class="text-sm text-gray-600">Téléchargez ou consultez vos billets</p>
</div>
<div class="space-y-6">
<!-- Download Tickets -->
<div class="flex items-start">
<div class="flex-shrink-0 w-8 h-8 bg-purple-100 rounded-full flex items-center justify-center">
<svg class="w-4 h-4 text-purple-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
</div>
<div class="ml-4">
<h3 class="font-semibold text-gray-900 mb-1">Télécharger Vos Billets</h3>
<p class="text-gray-600 text-sm mb-3">Gardez vos billets sur votre téléphone ou imprimez-les.</p>
<div class="space-y-2">
<% @tickets.each_with_index do |ticket, index| %>
<div class="flex items-center justify-between p-3 border border-purple-200 rounded-lg bg-purple-50 hover:bg-purple-100 transition-colors">
<%= link_to ticket_path(ticket.qr_code), class: "flex-1 flex items-center text-purple-700 hover:text-purple-800 font-medium" do %>
<div class="flex items-center justify-center w-6 h-6 bg-purple-200 text-purple-800 text-xs font-bold rounded-full mr-3">
<%= index + 1 %>
</div>
<%= ticket.first_name %> <%= ticket.last_name %>
<% end %>
<%= link_to ticket_download_path(ticket.qr_code), class: "ml-3 p-2 text-purple-600 hover:text-purple-800 hover:bg-purple-200 rounded-lg transition-colors", title: "Télécharger le billet PDF" do %>
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
<% end %>
</div>
<% end %>
</div>
</div>
<!-- Order Total -->
<div class="border-t border-gray-200 pt-6">
<div class="flex items-center justify-between text-lg">
<span class="font-medium text-gray-900">Total</span>
<span class="font-bold text-2xl text-purple-600"><%= @order.total_amount_euros %>€</span>
</div>
<p class="text-xs text-gray-500 mt-2">TVA incluse</p>
</div>
<!-- Actions -->
<div class="border-t border-gray-200 pt-6 mt-6">
<div class="flex space-x-4">
<%= link_to event_path(@order.event.slug, @order.event), class: "bg-gray-100 hover:bg-gray-200 text-gray-700 font-medium py-2 px-4 rounded-lg transition-colors" do %>
<div class="flex items-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10 19l-7-7m0 0l7-7m-7 7h18"/>
</svg>
Retour à l'événement
</div>
<!-- Event Day -->
<div class="flex items-start">
<div class="flex-shrink-0 w-8 h-8 bg-green-100 rounded-full flex items-center justify-center">
<svg class="w-4 h-4 text-green-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
</div>
<div class="ml-4">
<h3 class="font-semibold text-gray-900 mb-1">Le Jour de l'Événement</h3>
<p class="text-gray-600 text-sm">Présentez votre billet (QR code) à l'entrée. Arrivez un peu en avance !</p>
</div>
</div>
</div>
<% else %>
<!-- Payment Required -->
<div class="border-b border-gray-200 pb-6 mb-6">
<h2 class="text-xl font-bold text-gray-900 mb-2">Paiement Requis</h2>
<p class="text-sm text-gray-600">Votre commande nécessite un paiement</p>
</div>
<% end %>
<% if @order.can_retry_payment? %>
<div class="mb-6">
<%= link_to checkout_order_path(@order), class: "block w-full text-center py-3 px-4 bg-orange-600 hover:bg-orange-700 text-white font-medium rounded-lg transition-colors" do %>
<div class="flex items-center justify-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M3 10h18M7 15h1m4 0h1m-7 4h12a3 3 0 003-3V8a3 3 0 00-3-3H6a3 3 0 00-3 3v8a3 3 0 003 3z"/>
</svg>
Procéder au Paiement
</div>
<% end %>
</div>
<%= link_to checkout_order_path(@order), class: "bg-purple-600 hover:bg-purple-700 text-white font-medium py-2 px-4 rounded-lg transition-colors" do %>
<div class="flex items-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M3 10h18M7 15h1m4 0h1m-7 4h12a3 3 0 003-3V8a3 3 0 00-3-3H6a3 3 0 00-3 3v8a3 3 0 003 3z"/>
</svg>
Procéder au paiement
</div>
<% end %>
<% end %>
<% end %>
<!-- Navigation Actions -->
<div class="border-t border-gray-200 pt-6 mt-6">
<div class="space-y-3">
<%= link_to dashboard_path, class: "block w-full text-center py-3 px-4 bg-purple-600 hover:bg-purple-700 text-white font-medium rounded-lg transition-colors" do %>
<div class="flex items-center justify-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10 19l-7-7m0 0l7-7m-7 7h18"/>
</svg>
Retour au tableau de bord
</div>
<% end %>
<%= link_to event_path(@order.event.slug, @order.event), class: "block w-full text-center py-3 px-4 border border-gray-300 rounded-lg text-gray-700 hover:bg-gray-50 transition-colors" do %>
<div class="flex items-center justify-center">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z"/>
</svg>
Voir la page d'évenement
</div>
<% end %>
</div>
</div>
</div>
</div>

View File

@@ -1,69 +1,74 @@
<div class="container mx-auto px-4 sm:px-6 lg:px-8 py-4 sm:py-6">
<!-- Simplified header -->
<div class="my-6 sm:my-8">
<div class="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-4">
<div>
<h1 class="text-2xl sm:text-3xl font-bold text-slate-900 dark:text-slate-100">Mon tableau de bord</h1>
<p class="text-sm sm:text-base text-slate-600 dark:text-slate-400 mt-1">Gérez vos commandes et accédez à vos billets</p>
</div>
<div class="container mx-auto px-4 sm:px-6 lg:px-8 py-8">
<!-- Hero section with metrics -->
<div class="mt-4 mb-8">
<div class="flex items-center justify-between mb-6">
<h1 class="text-3xl font-bold text-slate-900 dark:text-slate-100">Tableau de bord</h1>
<!-- Promoter Actions -->
<% if current_user.promoter? %>
<div class="flex flex-col xs:flex-row items-stretch xs:items-center gap-2">
<%= link_to promoter_events_path, class: "inline-flex items-center justify-center px-3 py-2 sm:px-4 sm:py-2 bg-purple-600 text-white font-medium rounded-lg hover:bg-purple-700 transition-colors duration-200 text-sm" do %>
<i data-lucide="calendar-plus" class="w-4 h-4 mr-1 sm:mr-2"></i>
<span class="whitespace-nowrap">Mes Événements</span>
<div class="flex items-center space-x-3">
<%= link_to promoter_events_path, class: "inline-flex items-center px-4 py-2 bg-purple-600 text-white font-medium rounded-lg hover:bg-purple-700 transition-colors duration-200" do %>
<i data-lucide="calendar-plus" class="w-4 h-4 mr-2"></i>
Mes événements
<% end %>
<%= link_to new_promoter_event_path, class: "inline-flex items-center justify-center px-3 py-2 sm:px-4 sm:py-2 bg-black text-white font-medium rounded-lg hover:bg-gray-800 transition-colors duration-200 text-sm" do %>
<i data-lucide="plus" class="w-4 h-4 mr-1 sm:mr-2"></i>
<span class="whitespace-nowrap">Créer un Événement</span>
<%= link_to new_promoter_event_path, class: "inline-flex items-center px-4 py-2 bg-black text-white font-medium rounded-lg hover:bg-gray-800 transition-colors duration-200" do %>
<i data-lucide="plus" class="w-4 h-4 mr-2"></i>
Créer un événement
<% end %>
</div>
<% else %>
<%= link_to events_path, class: "inline-flex items-center justify-center px-3 py-2 sm:px-4 sm:py-2 bg-purple-600 text-white font-medium rounded-lg hover:bg-purple-700 transition-colors duration-200 text-sm" do %>
<i data-lucide="search" class="w-4 h-4 mr-1 sm:mr-2"></i>
<span class="whitespace-nowrap">Découvrir des Événements</span>
<% end %>
<% end %>
</div>
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-6">
<%= render partial: 'components/metric_card', locals: { title: "Mes réservations", value: @booked_events, classes: "from-green-100 to-emerald-100" } %>
<%= render partial: 'components/metric_card', locals: { title: "Événements aujourd'hui", value: @events_today, classes: "from-blue-100 to-sky-100" } %>
<%= render partial: 'components/metric_card', locals: { title: "Événements demain", value: @events_tomorrow, classes: "from-purple-100 to-indigo-100" } %>
<%= render partial: 'components/metric_card', locals: { title: "À venir", value: @upcoming_events, classes: "from-orange-100 to-amber-100" } %>
</div>
</div>
<!-- Draft orders needing payment -->
<% if @draft_orders.any? %>
<div class="card mb-6 sm:mb-8 border-orange-200 bg-orange-50">
<div class="card hover-lift mb-8 border-orange-200 bg-orange-50">
<div class="card-header bg-orange-100 rounded-lg">
<div class="mx-4 py-3 sm:py-4">
<h2 class="text-lg sm:text-2xl font-bold text-orange-900 flex items-center">
<svg class="w-5 h-5 sm:w-6 sm:h-6 mr-2 text-orange-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<div class="mx-4 py-4">
<h2 class="text-2xl font-bold text-orange-900 flex items-center">
<svg class="w-6 h-6 mr-2 text-orange-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4m0 4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
Commandes en Attente de Paiement
Commandes en attente de paiement
</h2>
<p class="text-sm sm:text-base text-orange-700 mt-1">Vous avez des commandes qui nécessitent un paiement</p>
<p class="text-orange-700 mt-1">Vous avez des commandes qui nécessitent un paiement</p>
</div>
</div>
<div class="card-body">
<div class="space-y-4">
<% @draft_orders.each do |order| %>
<div class="bg-white rounded-lg p-3 sm:p-4 border border-orange-200">
<div class="flex flex-col sm:flex-row sm:items-start sm:justify-between gap-3 mb-3">
<div class="bg-white rounded-lg p-4 border border-orange-200">
<div class="flex items-start justify-between mb-3">
<div>
<h3 class="font-semibold text-gray-900 text-sm sm:text-base"><%= order.event.name %></h3>
<p class="text-xs sm:text-sm text-gray-600 mt-1">
<svg class="w-3 h-3 sm:w-4 sm:h-4 inline mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<h3 class="font-semibold text-gray-900"><%= order.event.name %></h3>
<p class="text-sm text-gray-600">
<svg class="w-4 h-4 inline mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z"/>
</svg>
<%= order.event.start_time.strftime("%d %B %Y à %H:%M") %>
</p>
</div>
<span class="text-xs sm:text-sm font-medium text-orange-600 bg-orange-100 px-2 py-1 rounded-full whitespace-nowrap">
Order #<%= order.id %>
<span class="text-sm font-medium text-orange-600 bg-orange-100 px-2 py-1 rounded-full">
Commande #<%= order.id %>
</span>
</div>
<div class="grid gap-2 mb-4">
<% order.tickets.each do |ticket| %>
<div class="flex flex-col sm:flex-row sm:items-center justify-between text-xs sm:text-sm bg-gray-50 rounded p-2 gap-2">
<div class="flex items-center justify-between text-sm bg-gray-50 rounded p-2">
<div>
<span class="font-medium"><%= ticket.ticket_type.name %></span>
<span class="text-gray-600">- <%= ticket.first_name %> <%= ticket.last_name %></span>
@@ -75,21 +80,19 @@
<% end %>
</div>
<div class="flex flex-col sm:flex-row items-start sm:items-center justify-between gap-3">
<div class="text-xs sm:text-sm text-gray-600">
<div class="mb-1 sm:mb-0">
Tentatives: <%= order.payment_attempts %>/3
</div>
<div class="flex items-center justify-between">
<div class="text-sm text-gray-600">
Tentatives: <%= order.payment_attempts %>/3
<% if order.expiring_soon? %>
<span class="text-orange-600 font-medium">⚠️ Expire dans <%= time_ago_in_words(order.expires_at) %></span>
<span class="text-orange-600 font-medium ml-2">⚠️ Expire dans <%= time_ago_in_words(order.expires_at) %></span>
<% else %>
<span class="text-gray-500">Expire dans <%= time_ago_in_words(order.expires_at) %></span>
<span class="text-gray-500 ml-2">Expire dans <%= time_ago_in_words(order.expires_at) %></span>
<% end %>
</div>
<%= link_to retry_payment_order_path(order), method: :post,
class: "inline-flex items-center px-3 py-2 sm:px-4 sm:py-2 bg-orange-600 text-white text-xs sm:text-sm font-medium rounded-lg hover:bg-orange-700 transition-colors duration-200 whitespace-nowrap" do %>
Reprendre le Paiement (<%= order.total_amount_euros %>)
class: "inline-flex items-center px-4 py-2 bg-orange-600 text-white text-sm font-medium rounded-lg hover:bg-orange-700 transition-colors duration-200" do %>
Reprendre le paiement (<%= order.total_amount_euros %>)
<% end %>
</div>
</div>
@@ -99,148 +102,96 @@
</div>
<% end %>
<!-- User's Orders Section -->
<div class="card mb-6 sm:mb-8">
<!-- User's booked events -->
<div class="card hover-lift mb-8">
<div class="card-header">
<div class="flex items-center justify-between">
<h2 class="text-xl sm:text-2xl font-bold text-slate-900 dark:text-slate-100">Mes Commandes</h2>
<span class="text-xs sm:text-sm text-slate-600 dark:text-slate-400 bg-slate-100 dark:bg-slate-700 px-2 py-1 sm:px-3 sm:py-1 rounded-full">
<%= pluralize(@user_orders.count, 'commande') %>
</span>
</div>
<h2 class="text-2xl font-bold text-slate-900 dark:text-slate-100">Mes événements réservés</h2>
</div>
<div class="card-body">
<% if @user_orders.any? %>
<div class="space-y-4">
<% @user_orders.each do |order| %>
<div class="bg-white dark:bg-slate-800 border border-slate-200 dark:border-slate-700 rounded-lg p-3 sm:p-4 hover:shadow-md transition-shadow">
<div class="flex flex-col sm:flex-row sm:items-start sm:justify-between gap-3 mb-3">
<div class="flex-1">
<div class="flex flex-wrap items-center gap-2 mb-2">
<h3 class="font-semibold text-slate-900 dark:text-slate-100 text-sm sm:text-base"><%= order.event.name %></h3>
<span class="text-xs px-2 py-1 rounded-full <%= order.status == 'paid' ? 'bg-green-100 text-green-800 dark:bg-green-800 dark:text-green-100' : order.status == 'completed' ? 'bg-blue-100 text-blue-800 dark:bg-blue-800 dark:text-blue-100' : 'bg-yellow-100 text-yellow-800 dark:bg-yellow-800 dark:text-yellow-100' %>">
<%= order.status.humanize %>
</span>
</div>
<div class="flex flex-wrap items-center gap-3 text-xs sm:text-sm text-slate-600 dark:text-slate-400 mb-2">
<div class="flex items-center">
<i data-lucide="calendar" class="w-3 h-3 sm:w-4 sm:h-4 mr-1"></i>
<%= order.event.start_time.strftime("%d %B %Y à %H:%M") %>
</div>
<div class="flex items-center">
<i data-lucide="map-pin" class="w-3 h-3 sm:w-4 sm:h-4 mr-1"></i>
<%= order.event.venue_name %>
</div>
<div class="flex items-center">
<i data-lucide="shopping-bag" class="w-3 h-3 sm:w-4 sm:h-4 mr-1"></i>
<%= pluralize(order.tickets.count, 'billet') %>
</div>
</div>
<div class="text-xs sm:text-sm text-slate-500 dark:text-slate-400 mt-2">
Order #<%= order.id %> • <%= order.created_at.strftime("%m/%d/%Y") %> • €<%= order.total_amount_euros %>
</div>
</div>
<div class="flex items-center">
<%= link_to order_path(order),
class: "inline-flex items-center px-3 py-2 bg-purple-600 hover:bg-purple-700 text-white text-xs sm:text-sm font-medium rounded-lg transition-colors duration-200 whitespace-nowrap" do %>
<i data-lucide="eye" class="w-3 h-3 sm:w-4 sm:h-4 mr-1"></i>
Voir les Détails
<% end %>
</div>
</div>
<!-- Quick tickets preview -->
<div class="border-t border-slate-200 dark:border-slate-600 pt-3">
<div class="grid gap-2">
<% order.tickets.limit(3).each do |ticket| %>
<div class="flex flex-col sm:flex-row sm:items-center justify-between text-xs sm:text-sm bg-slate-50 dark:bg-slate-700 rounded p-2 gap-2">
<div class="flex items-center space-x-2">
<span class="w-2 h-2 bg-green-500 rounded-full"></span>
<span class="font-medium"><%= ticket.ticket_type.name %></span>
<span class="text-slate-500 text-xs">- <%= ticket.first_name %> <%= ticket.last_name %></span>
</div>
<div class="flex items-center space-x-2">
<%= link_to ticket_download_path(ticket.qr_code),
class: "text-purple-600 hover:text-purple-800 dark:text-purple-400 dark:hover:text-purple-200" do %>
<i data-lucide="download" class="w-3 h-3"></i>
<% end %>
</div>
</div>
<% end %>
<% if order.tickets.count > 3 %>
<div class="text-xs text-slate-500 text-center">
et <%= pluralize(order.tickets.count - 3, 'autre billet') %>
</div>
<% end %>
</div>
</div>
</div>
<% if @user_booked_events.any? %>
<ul class="space-y-4">
<% @user_booked_events.each do |event| %>
<li>
<%= render partial: 'components/event_item', locals: { event: event } %>
</li>
<% end %>
</div>
<% if @user_orders.count >= 10 %>
<div class="mt-4 sm:mt-6 text-center">
<%= link_to "Voir Toutes Mes Commandes", orders_path, class: "text-purple-600 dark:text-purple-400 hover:text-purple-800 dark:hover:text-purple-200 font-medium transition-colors duration-200 text-sm" %>
</ul>
<% if @booked_events > 5 %>
<div class="mt-6 text-center">
<%= link_to "Voir toutes mes réservations", "#", class: "text-purple-600 dark:text-purple-400 hover:text-purple-800 dark:hover:text-purple-200 font-medium transition-colors duration-200" %>
</div>
<% end %>
<% else %>
<div class="text-center py-8 sm:py-12">
<div class="w-12 h-12 sm:w-16 sm:h-16 bg-slate-100 dark:bg-slate-700 rounded-full flex items-center justify-center mx-auto mb-3 sm:mb-4">
<i data-lucide="shopping-bag" class="w-6 h-6 sm:w-8 sm:h-8 text-slate-400"></i>
</div>
<h3 class="text-base sm:text-lg font-medium text-slate-900 dark:text-slate-100 mb-1 sm:mb-2">Aucune Commande</h3>
<p class="text-sm sm:text-base text-slate-600 dark:text-slate-400 mb-4 sm:mb-6">Vous n'avez pas encore passé de commandes.</p>
<%= link_to events_path, class: "inline-flex items-center px-3 py-2 sm:px-4 sm:py-2 bg-purple-600 text-white rounded-lg hover:bg-purple-700 transition-colors duration-200 text-sm" do %>
<i data-lucide="search" class="w-3 h-3 sm:w-4 sm:h-4 mr-1 sm:mr-2"></i>
Découvrir des Événements
<% end %>
<div class="text-center py-8">
<p class="text-slate-600 dark:text-slate-400 mb-4">Vous n'avez encore réservé aucun événement.</p>
<%= link_to "Découvrir les événements", events_path, class: "inline-flex items-center px-4 py-2 bg-purple-600 text-white rounded-lg hover:bg-purple-700 transition-colors duration-200" %>
</div>
<% end %>
</div>
</div>
<!-- Quick Events Preview - Simplified -->
<% if @user_orders.any? %>
<div class="my-6 sm:my-8 card">
<div class="card-header">
<div class="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-3">
<h2 class="text-lg sm:text-xl font-bold text-slate-900 dark:text-slate-100">Découvrir d'autres événements</h2>
<%= link_to events_path, class: "text-purple-600 dark:text-purple-400 hover:text-purple-800 dark:hover:text-purple-200 font-medium text-sm transition-colors duration-200 whitespace-nowrap" do %>
Voir tout →
<% end %>
</div>
</div>
<div class="card-body">
<% if @upcoming_preview_events.any? %>
<div class="grid gap-3 sm:gap-4 grid-cols-1 sm:grid-cols-2 lg:grid-cols-3">
<% @upcoming_preview_events.each do |event| %>
<div class="bg-white dark:bg-slate-800 border border-slate-200 dark:border-slate-700 rounded-lg p-3 sm:p-4 hover:shadow-md transition-shadow">
<h4 class="font-medium text-slate-900 dark:text-slate-100 mb-2 text-sm sm:text-base"><%= event.name %></h4>
<div class="text-xs sm:text-sm text-slate-600 dark:text-slate-400 space-y-1">
<div class="flex items-center">
<i data-lucide="calendar" class="w-3 h-3 mr-1"></i>
<%= event.start_time.strftime("%d %B") %>
</div>
<div class="flex items-center">
<i data-lucide="map-pin" class="w-3 h-3 mr-1"></i>
<%= event.venue_name %>
</div>
</div>
<div class="mt-2 sm:mt-3">
<%= link_to event_path(event.slug, event), class: "text-purple-600 dark:text-purple-400 hover:text-purple-800 dark:hover:text-purple-200 text-xs sm:text-sm font-medium" do %>
Voir l'Événement →
<% end %>
</div>
</div>
<% end %>
</div>
<% else %>
<p class="text-slate-600 dark:text-slate-400 text-sm">Aucun événement à venir pour le moment.</p>
<% end %>
</div>
<!-- Today's events -->
<div class="card hover-lift mb-8">
<div class="card-header">
<h2 class="text-2xl font-bold text-slate-900 dark:text-slate-100">Évenements du jour</h2>
</div>
<% end %>
<div class="card-body">
<% if @today_events.any? %>
<ul class="space-y-4">
<% @today_events.each do |event| %>
<li>
<%= render partial: 'components/event_item', locals: { event: event } %>
</li>
<% end %>
</ul>
<% else %>
<p class="text-slate-600 dark:text-slate-400">Aucun évenement aujourd'hui.</p>
<% end %>
</div>
</div>
<!-- Tomorrow's events -->
<div class="card hover-lift mb-8">
<div class="card-header">
<h2 class="text-2xl font-bold text-slate-900 dark:text-slate-100">Évenements de demain</h2>
</div>
<div class="card-body">
<% if @tomorrow_events.any? %>
<ul class="space-y-4">
<% @tomorrow_events.each do |event| %>
<li>
<%= render partial: 'components/event_item', locals: { event: event } %>
</li>
<% end %>
</ul>
<% else %>
<p class="text-slate-600 dark:text-slate-400">Aucune partie demain.</p>
<% end %>
</div>
</div>
<!-- Other upcoming events with pagination -->
<div class="card hover-lift">
<div class="card-header">
<h2 class="text-2xl font-bold text-slate-900 dark:text-slate-100">Autres évenements à venir</h2>
</div>
<div class="card-body">
<% if @other_events.any? %>
<ul class="space-y-4">
<% @other_events.each do |event| %>
<li>
<%= render partial: 'components/event_item', locals: { event: event } %>
</li>
<% end %>
</ul>
<!-- Pagination -->
<div class="mt-8">
<%= paginate @other_events %>
</div>
<% else %>
<p class="text-slate-600 dark:text-slate-400">Aucune autre partie à venir.</p>
<% end %>
</div>
</div>
</div>

View File

@@ -1,7 +1,7 @@
<% if flash.any? %>
<div class="container mx-auto px-4 sm:px-6 lg:px-8">
<div class="relative w-full p-4 mt-4">
<div class="w-full">
<div class="container">
<div class="relative w-full flex justify-center p-4 mt-4">
<div class="w-full max-w-xl">
<% flash.each do |type, message| %>
<div class="notification <%= flash_class(type) %> flex items-center gap-3 p-4 rounded-lg mb-3 font-medium w-full box-border"
data-controller="flash-message">

View File

@@ -1,86 +0,0 @@
<div style="font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; max-width: 600px; margin: 0 auto; padding: 20px; background-color: #f8f9fa; border-radius: 8px;">
<div style="text-align: center; padding: 20px 0; border-bottom: 1px solid #e9ecef;">
<h1 style="color: #4c1d95; margin: 0; font-size: 28px;">ApéroNight</h1>
<p style="color: #6c757d; margin: 10px 0 0;">Rappel d'événement</p>
</div>
<div style="background-color: white; border-radius: 8px; padding: 30px; margin: 20px 0; box-shadow: 0 2px 4px rgba(0,0,0,0.1);">
<h2 style="color: #212529; margin-top: 0;">Salut <%= @user.email.split('@').first %> ! 🎉</h2>
<p style="color: #495057; line-height: 1.6; font-size: 18px;">
<% case @days_before %>
<% when 7 %>
Plus qu'une semaine avant <strong><%= @event.name %></strong> !
<% when 1 %>
C'est demain ! <strong><%= @event.name %></strong> a lieu demain.
<% when 0 %>
C'est aujourd'hui ! <strong><%= @event.name %></strong> a lieu aujourd'hui.
<% else %>
Plus que <%= @days_before %> jours avant <strong><%= @event.name %></strong> !
<% end %>
</p>
<div style="background-color: #f8f9fa; border-radius: 6px; padding: 20px; margin: 25px 0;">
<h3 style="color: #4c1d95; margin-top: 0; border-bottom: 1px solid #e9ecef; padding-bottom: 10px;">Détails de l'événement</h3>
<div style="display: flex; justify-content: space-between; margin-bottom: 15px;">
<div>
<p style="margin: 0; color: #6c757d; font-size: 14px;">📅 Date & heure</p>
<p style="margin: 5px 0 0; font-weight: bold; color: #212529; font-size: 16px;"><%= @event.start_time.strftime("%d %B %Y à %H:%M") %></p>
</div>
</div>
<div style="margin-bottom: 15px;">
<p style="margin: 0; color: #6c757d; font-size: 14px;">📍 Lieu</p>
<p style="margin: 5px 0 0; font-weight: bold; color: #212529;"><%= @event.venue_name %></p>
<p style="margin: 5px 0 0; color: #495057;"><%= @event.venue_address %></p>
</div>
</div>
<div style="background-color: #f8f9fa; border-radius: 6px; padding: 20px; margin: 25px 0;">
<h4 style="color: #4c1d95; margin-top: 0; margin-bottom: 15px;">Vos billets pour cet événement :</h4>
<% @tickets.each_with_index do |ticket, index| %>
<div style="border: 1px solid #e9ecef; border-radius: 4px; padding: 15px; margin-bottom: 10px; background-color: white;">
<div style="display: flex; justify-content: space-between; align-items: center;">
<div>
<p style="margin: 0 0 5px; font-weight: bold; color: #212529;">🎫 Billet #<%= index + 1 %></p>
<p style="margin: 0; color: #6c757d; font-size: 14px;"><%= ticket.ticket_type.name %></p>
<p style="margin: 5px 0 0;"><a href="<%= ticket_url(ticket) %>" style="color: #4c1d95; text-decoration: none; font-size: 14px;">📱 Voir le détail et le code QR</a></p>
</div>
<div style="text-align: right;">
<span style="background-color: #d4edda; color: #155724; padding: 4px 8px; border-radius: 4px; font-size: 12px; font-weight: bold;">ACTIF</span>
</div>
</div>
</div>
<% end %>
</div>
<div style="text-align: center; margin: 30px 0;">
<% if @days_before == 0 %>
<p style="color: #495057; margin-bottom: 20px; font-size: 16px;">🚨 N'oubliez pas vos billets ! Ils ont été envoyés par email lors de votre achat.</p>
<% else %>
<p style="color: #495057; margin-bottom: 20px;">📧 Vos billets ont été envoyés par email lors de votre achat.</p>
<% end %>
<p style="color: #495057; margin-bottom: 20px;">Présentez-les à l'entrée de l'événement pour y accéder.</p>
</div>
<% if @days_before <= 1 %>
<div style="background-color: #d1ecf1; border-radius: 6px; padding: 15px; border-left: 4px solid #17a2b8; margin: 20px 0;">
<p style="margin: 0; color: #0c5460; font-size: 14px;">
<strong>💡 Conseil :</strong> Arrivez un peu en avance pour éviter les files d'attente à l'entrée !
</p>
</div>
<% else %>
<div style="background-color: #d4edda; border-radius: 6px; padding: 15px; border-left: 4px solid #28a745;">
<p style="margin: 0; color: #155724; font-size: 14px;">
<strong>📅 Ajoutez à votre calendrier :</strong> N'oubliez pas d'ajouter cet événement à votre calendrier pour ne pas le manquer !
</p>
</div>
<% end %>
</div>
<div style="text-align: center; color: #6c757d; font-size: 14px; padding: 20px 0;">
<p style="margin: 0;">Des questions ? Contactez-nous à <a href="mailto:support@aperonight.com" style="color: #4c1d95; text-decoration: none;">support@aperonight.com</a></p>
<p style="margin: 10px 0 0;">&copy; <%= Time.current.year %> ApéroNight. Tous droits réservés.</p>
</div>
</div>

View File

@@ -1,41 +0,0 @@
Salut <%= @user.email.split('@').first %> !
<% case @days_before %>
<% when 7 %>
Plus qu'une semaine avant "<%= @event.name %>" !
<% when 1 %>
C'est demain ! "<%= @event.name %>" a lieu demain.
<% when 0 %>
C'est aujourd'hui ! "<%= @event.name %>" a lieu aujourd'hui.
<% else %>
Plus que <%= @days_before %> jours avant "<%= @event.name %>" !
<% end %>
DÉTAILS DE L'ÉVÉNEMENT
======================
Date & heure : <%= @event.start_time.strftime("%d %B %Y à %H:%M") %>
Lieu : <%= @event.venue_name %>
Adresse : <%= @event.venue_address %>
VOS BILLETS POUR CET ÉVÉNEMENT :
<% @tickets.each_with_index do |ticket, index| %>
- Billet #<%= index + 1 %> : <%= ticket.ticket_type.name %> (ACTIF)
<% end %>
<% if @days_before == 0 %>
N'oubliez pas vos billets ! Ils ont été envoyés par email lors de votre achat.
<% else %>
Vos billets ont été envoyés par email lors de votre achat.
<% end %>
Présentez-les à l'entrée de l'événement pour y accéder.
<% if @days_before <= 1 %>
Conseil : Arrivez un peu en avance pour éviter les files d'attente à l'entrée !
<% else %>
N'oubliez pas d'ajouter cet événement à votre calendrier pour ne pas le manquer !
<% end %>
Des questions ? Contactez-nous à support@aperonight.com
© <%= Time.current.year %> ApéroNight. Tous droits réservés.

View File

@@ -1,116 +1,50 @@
<div style="font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; max-width: 600px; margin: 0 auto; padding: 20px; background-color: #f8f9fa; border-radius: 8px;">
<div style="text-align: center; padding: 20px 0; border-bottom: 1px solid #e9ecef;">
<h1 style="color: #4c1d95; margin: 0; font-size: 28px;"><%= ENV.fetch("APP_NAME", "Aperonight") %></h1>
<h1 style="color: #4c1d95; margin: 0; font-size: 28px;">ApéroNight</h1>
<p style="color: #6c757d; margin: 10px 0 0;">Confirmation de votre achat</p>
</div>
<div style="background-color: white; border-radius: 8px; padding: 30px; margin: 20px 0; box-shadow: 0 2px 4px rgba(0,0,0,0.1);">
<% if user.first_name %>
<h2 style="color: #212529; margin-top: 0;">Bonjour <%= @user.first_name %>,</h2>
<% else %>
<h2 style="color: #212529; margin-top: 0;">Bonjour <%= @user.email.split('@').first %>,</h2>
<% end %>
<h2 style="color: #212529; margin-top: 0;">Bonjour <%= @user.email.split('@').first %>,</h2>
<p style="color: #495057; line-height: 1.6;">
<% if defined?(@order) && @order.present? %>
Merci pour votre achat ! Nous avons le plaisir de vous confirmer votre commande pour l'événement <strong><%= @event.name %></strong>.
<% else %>
Merci pour votre achat ! Nous avons le plaisir de vous confirmer votre billet pour l'événement <strong><%= @event.name %></strong>.
<% end %>
Merci pour votre achat ! Nous avons le plaisir de vous confirmer votre billet pour l'événement <strong><%= @event.name %></strong>.
</p>
<div style="background-color: #f8f9fa; border-radius: 6px; padding: 20px; margin: 25px 0;">
<% if defined?(@order) && @order.present? %>
<h3 style="color: #4c1d95; margin-top: 0; border-bottom: 1px solid #e9ecef; padding-bottom: 10px;">Détails de votre commande</h3>
<h3 style="color: #4c1d95; margin-top: 0; border-bottom: 1px solid #e9ecef; padding-bottom: 10px;">Détails de votre billet</h3>
<div style="margin-bottom: 20px;">
<div style="display: flex; justify-content: space-between; margin-bottom: 15px;">
<div>
<p style="margin: 0; color: #6c757d; font-size: 14px;">Événement</p>
<p style="margin: 5px 0 0; font-weight: bold; color: #212529;"><%= @event.name %></p>
</div>
<div style="text-align: right;">
<p style="margin: 0; color: #6c757d; font-size: 14px;">Date & heure</p>
<p style="margin: 5px 0 0; font-weight: bold; color: #212529;"><%= @event.start_time.strftime("%d %B %Y à %H:%M") %></p>
</div>
</div>
<div style="display: flex; justify-content: space-between;">
<div>
<p style="margin: 0; color: #6c757d; font-size: 14px;">Nombre de billets</p>
<p style="margin: 5px 0 0; font-weight: bold; color: #212529;"><%= @tickets.count %></p>
</div>
<div style="text-align: right;">
<p style="margin: 0; color: #6c757d; font-size: 14px;">Total</p>
<p style="margin: 5px 0 0; font-weight: bold; color: #212529;"><%= number_to_currency(@order.total_amount_euros, unit: "€") %></p>
</div>
</div>
<div style="display: flex; justify-content: space-between; margin-bottom: 15px;">
<div>
<p style="margin: 0; color: #6c757d; font-size: 14px;">Événement</p>
<p style="margin: 5px 0 0; font-weight: bold; color: #212529;"><%= @event.name %></p>
</div>
<h4 style="color: #4c1d95; margin: 20px 0 15px;">Billets inclus :</h4>
<% @tickets.each_with_index do |ticket, index| %>
<div style="border: 1px solid #e9ecef; border-radius: 4px; padding: 15px; margin-bottom: 10px; background-color: white;">
<div style="display: flex; justify-content: space-between;">
<div>
<p style="margin: 0 0 5px; font-weight: bold; color: #212529;">Billet #<%= index + 1 %></p>
<p style="margin: 0; color: #6c757d; font-size: 14px;"><%= ticket.ticket_type.name %></p>
<p style="margin: 5px 0 0;"><a href="<%= ticket_url(ticket) %>" style="color: #4c1d95; text-decoration: none; font-size: 14px;">📱 Voir le détail et le code QR</a></p>
</div>
<div style="text-align: right;">
<p style="margin: 0; font-weight: bold; color: #212529;"><%= number_to_currency(ticket.price_cents / 100.0, unit: "€") %></p>
</div>
</div>
</div>
<% end %>
<% else %>
<h3 style="color: #4c1d95; margin-top: 0; border-bottom: 1px solid #e9ecef; padding-bottom: 10px;">Détails de votre billet</h3>
<div style="display: flex; justify-content: space-between; margin-bottom: 15px;">
<div>
<p style="margin: 0; color: #6c757d; font-size: 14px;">Événement</p>
<p style="margin: 5px 0 0; font-weight: bold; color: #212529;"><%= @event.name %></p>
</div>
<div style="text-align: right;">
<p style="margin: 0; color: #6c757d; font-size: 14px;">Type de billet</p>
<p style="margin: 5px 0 0; font-weight: bold; color: #212529;"><%= @ticket.ticket_type.name %></p>
</div>
<div style="text-align: right;">
<p style="margin: 0; color: #6c757d; font-size: 14px;">Type de billet</p>
<p style="margin: 5px 0 0; font-weight: bold; color: #212529;"><%= @ticket.ticket_type.name %></p>
</div>
</div>
<div style="display: flex; justify-content: space-between;">
<div>
<p style="margin: 0; color: #6c757d; font-size: 14px;">Date & heure</p>
<p style="margin: 5px 0 0; font-weight: bold; color: #212529;"><%= @event.start_time.strftime("%d %B %Y à %H:%M") %></p>
</div>
<div style="text-align: right;">
<p style="margin: 0; color: #6c757d; font-size: 14px;">Prix</p>
<p style="margin: 5px 0 0; font-weight: bold; color: #212529;"><%= number_to_currency(@ticket.price_cents / 100.0, unit: "€") %></p>
</div>
<div style="display: flex; justify-content: space-between;">
<div>
<p style="margin: 0; color: #6c757d; font-size: 14px;">Date & heure</p>
<p style="margin: 5px 0 0; font-weight: bold; color: #212529;"><%= @event.start_time.strftime("%d %B %Y à %H:%M") %></p>
</div>
<div style="margin-top: 15px; text-align: center;">
<a href="<%= ticket_url(@ticket) %>" style="color: #4c1d95; text-decoration: none; font-size: 14px; display: inline-block; padding: 10px 15px; border: 1px solid #4c1d95; border-radius: 6px; background-color: #f8f9fa;">📱 Voir le détail et le code QR</a>
<div style="text-align: right;">
<p style="margin: 0; color: #6c757d; font-size: 14px;">Prix</p>
<p style="margin: 5px 0 0; font-weight: bold; color: #212529;"><%= number_to_currency(@ticket.price_cents / 100.0, unit: "€") %></p>
</div>
<% end %>
</div>
</div>
<div style="text-align: center; margin: 30px 0;">
<% if defined?(@order) && @order.present? %>
<p style="color: #495057; margin-bottom: 20px;">Vos billets sont attachés à cet email en format PDF.</p>
<p style="color: #495057; margin-bottom: 20px;">Présentez-les à l'entrée de l'événement pour y accéder.</p>
<% else %>
<p style="color: #495057; margin-bottom: 20px;">Votre billet est attaché à cet email en format PDF.</p>
<p style="color: #495057; margin-bottom: 20px;">Présentez-le à l'entrée de l'événement pour y accéder.</p>
<% end %>
<p style="color: #495057; margin-bottom: 20px;">Votre billet est attaché à cet email en format PDF.</p>
<p style="color: #495057; margin-bottom: 20px;">Présentez-le à l'entrée de l'événement pour y accéder.</p>
</div>
<div style="background-color: #fff3cd; border-radius: 6px; padding: 15px; border-left: 4px solid #ffc107;">
<p style="margin: 0; color: #856404; font-size: 14px;">
<strong>Important :</strong>
<% if defined?(@order) && @order.present? %>
Ces billets sont valables pour une seule entrée chacun. Conservez-les précieusement.
<% else %>
Ce billet est valable pour une seule entrée. Conservez-le précieusement.
<% end %>
<strong>Important :</strong> Ce billet est valable pour une seule entrée. Conservez-le précieusement.
</p>
</div>
</div>

View File

@@ -1,29 +1,5 @@
<% if @user.first_name %>
Bonjour <%= @user.first_name %>,
<% else %>
Bonjour <%= @user.email.split('@').first %>,
<% end %>
Bonjour <%= @user.email.split('@').first %>,
<% if defined?(@order) && @order.present? %>
Merci pour votre achat ! Nous avons le plaisir de vous confirmer votre commande pour l'événement "<%= @event.name %>".
DÉTAILS DE VOTRE COMMANDE
=========================
Événement : <%= @event.name %>
Date & heure : <%= @event.start_time.strftime("%d %B %Y à %H:%M") %>
Nombre de billets : <%= @tickets.count %>
Total : <%= number_to_currency(@order.total_amount_euros, unit: "€") %>
BILLETS INCLUS :
<% @tickets.each_with_index do |ticket, index| %>
- Billet #<%= index + 1 %> : <%= ticket.ticket_type.name %> - <%= number_to_currency(ticket.price_cents / 100.0, unit: "€") %>
<% end %>
Vos billets sont attachés à cet email en format PDF. Présentez-les à l'entrée de l'événement pour y accéder.
Important : Ces billets sont valables pour une seule entrée chacun. Conservez-les précieusement.
<% else %>
Merci pour votre achat ! Nous avons le plaisir de vous confirmer votre billet pour l'événement "<%= @event.name %>".
DÉTAILS DE VOTRE BILLET
@@ -37,8 +13,7 @@ Prix : <%= number_to_currency(@ticket.price_cents / 100.0, unit: "€") %>
Votre billet est attaché à cet email en format PDF. Présentez-le à l'entrée de l'événement pour y accéder.
Important : Ce billet est valable pour une seule entrée. Conservez-le précieusement.
<% end %>
Si vous avez des questions, contactez-nous à support@aperonight.com
© <%= Time.current.year %> <%= ENV.fetch("APP_NAME", "Aperonight") %>. Tous droits réservés.
© <%= Time.current.year %> ApéroNight. Tous droits réservés.

View File

@@ -0,0 +1,98 @@
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>Ticket #<%= ticket.id %></title>
<style>
body {
font-family: Helvetica, Arial, sans-serif;
font-size: 12px;
color: #000000;
margin: 0;
padding: 20px;
background-color: #ffffff;
}
.ticket-container {
max-width: 350px;
margin: 0 auto;
padding: 20px;
border: 1px solid #e5e7eb;
border-radius: 10px;
background-color: #ffffff;
}
.header {
text-align: center;
margin-bottom: 20px;
}
.header h1 {
color: #2D1B69;
font-size: 24px;
font-weight: bold;
margin: 0;
}
.event-name {
text-align: center;
margin-bottom: 20px;
}
.event-name h2 {
color: #000000;
font-size: 18px;
font-weight: bold;
margin: 0;
}
.ticket-info {
margin-bottom: 20px;
}
.info-row {
margin-bottom: 8px;
font-size: 14px;
}
.qr-code-section {
text-align: center;
margin-top: 20px;
}
.qr-code-container svg {
width: 120px;
height: 120px;
}
</style>
</head>
<body>
<div class="ticket-container">
<div class="header">
<h1>ApéroNight</h1>
</div>
<div class="event-name">
<h2><%= ticket.event.name %></h2>
</div>
<div class="ticket-info">
<div class="info-row">
<strong>Ticket Holder:</strong> <%= ticket.first_name %> <%= ticket.last_name %>
</div>
<div class="info-row">
<strong>Ticket Type:</strong> <%= ticket.ticket_type.name %>
</div>
<div class="info-row">
<strong>Price:</strong> €<%= ticket.price_euros %>
</div>
</div>
<div class="qr-code-section">
<div class="qr-code-container">
<%= raw ticket.generate_qr_svg %>
</div>
</div>
</div>
</body>
</html>

View File

@@ -84,7 +84,7 @@
</div>
<div class="flex items-center space-x-2">
<%= link_to ticket_download_path(ticket.qr_code, format: :pdf),
<%= link_to download_ticket_path(ticket, format: :pdf),
class: "inline-flex items-center px-4 py-2 bg-gradient-to-r from-purple-600 to-indigo-600 text-white rounded-lg hover:from-purple-700 hover:to-indigo-700 transition-all duration-200 text-sm font-medium shadow-sm" do %>
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"></path>

View File

@@ -1,55 +1,44 @@
<div class="min-h-screen bg-gradient-to-br from-gray-50 to-gray-100 py-8">
<div class="min-h-screen bg-gradient-to-br from-slate-50 to-slate-100 py-8">
<div class="max-w-4xl mx-auto px-4 sm:px-6 lg:px-8">
<!-- Breadcrumb -->
<nav class="flex mb-6" aria-label="Breadcrumb">
<ol class="inline-flex items-center space-x-1 md:space-x-2 rounded-lg bg-white px-4 py-2 shadow-sm">
<li class="inline-flex items-center">
<%= link_to "Accueil", root_path, class: "inline-flex items-center text-sm font-medium text-gray-700 hover:text-purple-600" %>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<%= link_to "Tableau de bord", dashboard_path, class: "ml-1 text-sm font-medium text-gray-700 hover:text-purple-600 md:ml-2" %>
</div>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<%= link_to "Commande ##{@order.id}", order_path(@order), class: "ml-1 text-sm font-medium text-gray-700 hover:text-purple-600 md:ml-2" %>
</div>
</li>
<li>
<div class="flex items-center">
<svg class="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z" clip-rule="evenodd"></path>
</svg>
<span class="ml-1 text-sm font-medium text-purple-600 md:ml-2">Billet #<%= @ticket.id %></span>
</div>
</li>
<nav class="mb-8" aria-label="Breadcrumb">
<ol class="flex items-center space-x-2 text-sm">
<%= link_to root_path, class: "text-slate-500 hover:text-purple-600 transition-colors duration-200" do %>
<svg class="w-4 h-4 inline-block mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2">
<path stroke-linecap="round" stroke-linejoin="round" d="M3 7v10a2 2 0 002 2h14a2 2 0 002-2V9a2 2 0 00-2-2h-6l-2-2H5a2 2 0 00-2 2z"/>
</svg>
Accueil
<% end %>
<svg class="w-4 h-4 text-slate-400" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2">
<path stroke-linecap="round" stroke-linejoin="round" d="M9 5l7 7-7 7"/>
</svg>
<%= link_to dashboard_path, class: "text-slate-500 hover:text-purple-600 transition-colors duration-200" do %>
Tableau de bord
<% end %>
<svg class="w-4 h-4 text-slate-400" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2">
<path stroke-linecap="round" stroke-linejoin="round" d="M9 5l7 7-7 7"/>
</svg>
<li class="font-medium text-slate-900" aria-current="page">Billet #<%= @ticket.id %></li>
</ol>
</nav>
<div class="bg-white rounded-2xl shadow-xl overflow-hidden">
<div class="bg-white rounded-2xl shadow-xl overflow-hidden border border-slate-200">
<!-- Ticket Header -->
<div class="bg-gradient-to-r from-purple-600 to-indigo-600 px-8 py-6">
<div class="bg-gradient-to-r from-purple-600 to-violet-600 px-8 py-6">
<div class="flex items-center justify-between">
<div>
<h1 class="text-2xl md:text-3xl font-bold text-white mb-2">Billet électronique</h1>
<h1 class="text-2xl md:text-3xl font-bold text-white mb-2">Billet Électronique</h1>
<p class="text-purple-100">ID: #<%= @ticket.id %></p>
</div>
<div class="text-right">
<div class="inline-flex items-center px-3 py-1 rounded-full text-sm font-medium <%=
case @ticket.status
when 'active' then 'bg-green-100 text-green-800'
when 'draft' then 'bg-yellow-100 text-yellow-800'
when 'used' then 'bg-gray-100 text-gray-800'
when 'active' then 'bg-emerald-100 text-emerald-800'
when 'draft' then 'bg-amber-100 text-amber-800'
when 'used' then 'bg-slate-100 text-slate-800'
when 'expired' then 'bg-red-100 text-red-800'
when 'refunded' then 'bg-blue-100 text-blue-800'
else 'bg-gray-100 text-gray-800'
when 'refunded' then 'bg-sky-100 text-sky-800'
else 'bg-slate-100 text-slate-800'
end %>">
<%=
case @ticket.status
@@ -69,47 +58,49 @@
<div class="grid grid-cols-1 lg:grid-cols-2 gap-8">
<!-- Event Details -->
<div>
<h2 class="text-xl font-semibold text-gray-900 mb-6">Détails de l'événement</h2>
<h2 class="text-xl font-semibold text-slate-900 mb-6">Détails de l'événement</h2>
<div class="space-y-4">
<div class="space-y-6">
<div>
<label class="block text-sm font-medium text-gray-500 mb-1">Événement</label>
<p class="text-lg font-semibold text-gray-900"><%= @event.name %></p>
<label class="block text-sm font-medium text-slate-500 mb-2">Événement</label>
<p class="text-lg font-semibold text-slate-900"><%= @event.name %></p>
</div>
<div class="grid grid-cols-1 md:grid-cols-2 gap-4">
<div>
<label class="block text-sm font-medium text-gray-500 mb-1">Date et heure</label>
<div class="flex items-center text-gray-900">
<svg class="w-4 h-4 mr-2 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z"/>
<label class="block text-sm font-medium text-slate-500 mb-2">Date et heure</label>
<div class="flex items-start text-slate-900">
<svg class="w-4 h-4 mr-2 mt-0.5 text-slate-400" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2">
<path stroke-linecap="round" stroke-linejoin="round" d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z"/>
</svg>
<%= @event.start_time.strftime("%d %B %Y") %><br>
<small class="text-gray-600"><%= @event.start_time.strftime("%H:%M") %></small>
<div>
<div class="font-medium"><%= @event.start_time.strftime("%d %B %Y") %></div>
<div class="text-sm text-slate-600"><%= @event.start_time.strftime("%H:%M") %></div>
</div>
</div>
</div>
<div>
<label class="block text-sm font-medium text-gray-500 mb-1">Lieu</label>
<div class="flex items-center text-gray-900">
<svg class="w-4 h-4 mr-2 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M17.657 16.657L13.414 20.9a1.998 1.998 0 01-2.827 0l-4.244-4.243a8 8 0 1111.314 0z"/>
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 11a3 3 0 11-6 0 3 3 0 016 0z"/>
<label class="block text-sm font-medium text-slate-500 mb-2">Lieu</label>
<div class="flex items-center text-slate-900">
<svg class="w-4 h-4 mr-2 text-slate-400" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2">
<path stroke-linecap="round" stroke-linejoin="round" d="M17.657 16.657L13.414 20.9a1.998 1.998 0 01-2.827 0l-4.244-4.243a8 8 0 1111.314 0z"/>
<path stroke-linecap="round" stroke-linejoin="round" d="M15 11a3 3 0 11-6 0 3 3 0 016 0z"/>
</svg>
<%= @event.venue_name %>
<span class="font-medium"><%= @event.venue_name %></span>
</div>
</div>
</div>
<div>
<label class="block text-sm font-medium text-gray-500 mb-1">Type de billet</label>
<p class="text-gray-900 font-medium"><%= @ticket.ticket_type.name %></p>
<p class="text-sm text-gray-600"><%= @ticket.ticket_type.description %></p>
<label class="block text-sm font-medium text-slate-500 mb-2">Type de billet</label>
<p class="text-slate-900 font-medium mb-1"><%= @ticket.ticket_type.name %></p>
<p class="text-sm text-slate-600"><%= @ticket.ticket_type.description %></p>
</div>
<div>
<label class="block text-sm font-medium text-gray-500 mb-1">Prix</label>
<p class="text-xl font-bold text-gray-900">
<label class="block text-sm font-medium text-slate-500 mb-2">Prix</label>
<p class="text-2xl font-bold text-slate-900">
<%= number_to_currency(@ticket.price_euros, unit: "€", separator: ",", delimiter: " ", format: "%n %u") %>
</p>
</div>
@@ -118,40 +109,36 @@
<!-- Ticket Details -->
<div>
<h2 class="text-xl font-semibold text-gray-900 mb-6">Informations du billet</h2>
<h2 class="text-xl font-semibold text-slate-900 mb-6">Informations du billet</h2>
<div class="space-y-4">
<div class="space-y-6">
<div class="grid grid-cols-1 md:grid-cols-2 gap-4">
<div>
<label class="block text-sm font-medium text-gray-500 mb-1">Prénom</label>
<p class="text-gray-900 font-medium"><%= @ticket.first_name %></p>
<label class="block text-sm font-medium text-slate-500 mb-2">Prénom</label>
<p class="text-slate-900 font-medium"><%= @ticket.first_name %></p>
</div>
<div>
<label class="block text-sm font-medium text-gray-500 mb-1">Nom</label>
<p class="text-gray-900 font-medium"><%= @ticket.last_name %></p>
<label class="block text-sm font-medium text-slate-500 mb-2">Nom</label>
<p class="text-slate-900 font-medium"><%= @ticket.last_name %></p>
</div>
</div>
<div>
<label class="block text-sm font-medium text-gray-500 mb-1">Date d'achat</label>
<p class="text-gray-900"><%= @ticket.created_at.strftime("%d %B %Y à %H:%M") %></p>
<label class="block text-sm font-medium text-slate-500 mb-2">Date d'achat</label>
<p class="text-slate-900 font-medium"><%= @ticket.created_at.strftime("%d %B %Y à %H:%M") %></p>
</div>
<div>
<label class="block text-sm font-medium text-gray-500 mb-1">QR Code</label>
<div class="bg-gray-50 rounded-lg p-4 text-center">
<div class="inline-block bg-white p-4 rounded-lg shadow-sm">
<div data-controller="qr-code" data-qr-code-data-value="<%= @ticket.qr_code %>" class="w-32 h-32">
<!-- Loading indicator -->
<div data-qr-code-target="loading" class="w-32 h-32 bg-gray-100 rounded flex items-center justify-center">
<div class="animate-spin rounded-full h-8 w-8 border-b-2 border-purple-600"></div>
</div>
<!-- QR code container -->
<div data-qr-code-target="container" class="w-32 h-32"></div>
<label class="block text-sm font-medium text-slate-500 mb-2">QR Code</label>
<div class="bg-slate-50 rounded-xl p-6 text-center border border-slate-200">
<div class="inline-block bg-white p-4 rounded-xl shadow-sm border border-slate-200">
<div class="w-64 h-64 flex items-center justify-center">
<%= raw @ticket.generate_qr_svg %>
</div>
</div>
<p class="text-xs text-gray-500 mt-2 font-mono"><%= @ticket.qr_code %></p>
<p class="text-xs text-slate-500 mt-3 font-mono tracking-wider"><%= @ticket.qr_code[0..7]... %></p>
<p class="text-xs text-slate-400 mt-1">Scannez ce code à l'entrée</p>
</div>
</div>
</div>
@@ -159,21 +146,21 @@
</div>
<!-- Actions -->
<div class="mt-8 pt-6 border-t border-gray-200">
<div class="mt-8 pt-6 border-t border-slate-200">
<div class="flex flex-col sm:flex-row gap-4">
<%= link_to order_path(@order),
class: "px-6 py-3 border border-gray-300 text-gray-700 rounded-xl hover:bg-gray-50 text-center font-medium transition-colors duration-200" do %>
<svg class="w-4 h-4 inline-block mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M7 16l-4-4m0 0l4-4m-4 4h18"/>
<%= link_to dashboard_path,
class: "flex items-center justify-center px-6 py-3 border border-slate-300 text-slate-700 rounded-xl hover:bg-slate-50 hover:border-slate-400 font-medium transition-all duration-200" do %>
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2">
<path stroke-linecap="round" stroke-linejoin="round" d="M7 16l-4-4m0 0l4-4m-4 4h18"/>
</svg>
Retour aux informations de commande
Retour au tableau de bord
<% end %>
<% if @ticket.status == 'active' %>
<%= link_to ticket_download_path(@ticket.qr_code),
class: "flex-1 bg-gradient-to-r from-purple-600 to-indigo-600 hover:from-purple-700 hover:to-indigo-700 text-white font-medium py-3 px-6 rounded-xl shadow-sm transition-all duration-200 focus:outline-none focus:ring-2 focus:ring-purple-500 focus:ring-offset-2 transform hover:-translate-y-0.5 text-center" do %>
<svg class="w-4 h-4 inline-block mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
<%= link_to download_ticket_path(@ticket.id),
class: "flex-1 flex items-center justify-center bg-gradient-to-r from-purple-600 to-violet-600 hover:from-purple-700 hover:to-violet-700 text-white font-medium py-3 px-6 rounded-xl shadow-sm hover:shadow-md transition-all duration-200 focus:outline-none focus:ring-2 focus:ring-purple-500 focus:ring-offset-2 transform hover:-translate-y-0.5" do %>
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2">
<path stroke-linecap="round" stroke-linejoin="round" d="M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
Télécharger le PDF
<% end %>
@@ -182,17 +169,26 @@
</div>
<!-- Important Notice -->
<div class="mt-6 bg-blue-50 border border-blue-200 rounded-lg p-4">
<div class="mt-6 bg-sky-50 border border-sky-200 rounded-xl p-6">
<div class="flex items-start">
<svg class="w-5 h-5 text-blue-600 mr-2 mt-0.5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"/>
<svg class="w-5 h-5 text-sky-600 mr-3 mt-0.5 flex-shrink-0" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2">
<path stroke-linecap="round" stroke-linejoin="round" d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
<div class="flex-1">
<h3 class="text-blue-800 font-medium mb-1">Informations importantes</h3>
<ul class="text-blue-700 text-sm space-y-1">
<li>• Présentez ce billet (ou son code QR) à l'entrée de l'événement</li>
<li>• Arrivez en avance pour éviter les files d'attente</li>
<li>• En cas de problème, contactez l'organisateur</li>
<h3 class="text-sky-800 font-semibold mb-2">Informations importantes</h3>
<ul class="text-sky-700 text-sm space-y-2">
<li class="flex items-start">
<span class="w-1.5 h-1.5 bg-sky-600 rounded-full mt-2 mr-3 flex-shrink-0"></span>
Présentez ce billet (ou son code QR) à l'entrée de l'événement
</li>
<li class="flex items-start">
<span class="w-1.5 h-1.5 bg-sky-600 rounded-full mt-2 mr-3 flex-shrink-0"></span>
Arrivez en avance pour éviter les files d'attente
</li>
<li class="flex items-start">
<span class="w-1.5 h-1.5 bg-sky-600 rounded-full mt-2 mr-3 flex-shrink-0"></span>
En cas de problème, contactez l'organisateur
</li>
</ul>
</div>
</div>

View File

@@ -0,0 +1,14 @@
<% content_for :title, "Ticket ##{ticket.id}" %>
<div style="font-family: Arial, sans-serif; max-width: 350px; margin: 20px auto; padding: 20px; border: 1px solid #ccc;">
<div style="text-align: center;">
<h1 style="color: #2D1B69;">ApéroNight</h1>
</div>
<h2><%= ticket.event.name %></h2>
<p>Ticket Holder: <%= ticket.first_name %> <%= ticket.last_name %></p>
<p>Ticket Type: <%= ticket.ticket_type.name %></p>
<p>Price: €<%= ticket.price_euros %></p>
<div style="text-align: center; margin-top: 20px;">
<%= raw ticket.generate_qr_svg %>
</div>
</div>

View File

@@ -0,0 +1,118 @@
<% content_for :title, "Billet ##{@ticket.id} - #{@ticket.event.name}" %>
<div class="min-h-screen bg-slate-100 py-8">
<div class="max-w-md mx-auto px-4">
<!-- Ticket Card -->
<div class="max-w-md bg-white rounded-xl shadow-2xl overflow-hidden mx-auto border border-slate-200">
<!-- Header -->
<div class="bg-gradient-to-r from-purple-700 to-violet-600 text-center py-6 px-6">
<h1 class="text-2xl font-bold text-white mb-2">ApéroNight</h1>
<div class="w-16 h-0.5 bg-purple-200 mx-auto rounded-full"></div>
</div>
<!-- Event Name -->
<div class="text-center py-4 px-6 bg-purple-50 border-b border-purple-100">
<h2 class="text-xl font-bold text-slate-900 leading-tight"><%= @ticket.event.name %></h2>
</div>
<!-- Ticket Information -->
<div class="p-6 space-y-4">
<!-- Ticket Holder -->
<div class="flex justify-between items-center py-2 border-b border-slate-100">
<span class="text-sm font-medium text-slate-600">Porteur du billet:</span>
<span class="text-sm font-semibold text-slate-900 text-right"><%= @ticket.first_name %> <%= @ticket.last_name %></span>
</div>
<!-- Ticket Type -->
<div class="flex justify-between items-center py-2 border-b border-slate-100">
<span class="text-sm font-medium text-slate-600">Type de billet:</span>
<span class="text-sm font-semibold text-slate-900"><%= @ticket.ticket_type.name %></span>
</div>
<!-- Price -->
<div class="flex justify-between items-center py-2 border-b border-slate-100">
<span class="text-sm font-medium text-slate-600">Prix:</span>
<span class="text-sm font-semibold text-slate-900">
<%= number_to_currency(@ticket.price_euros, unit: "€", separator: ",", delimiter: " ", format: "%n %u") %>
</span>
</div>
<!-- Date & Time -->
<div class="flex justify-between items-center py-2 border-b border-slate-100">
<span class="text-sm font-medium text-slate-600">Date & Heure:</span>
<div class="text-right">
<div class="text-sm font-semibold text-slate-900"><%= @ticket.event.start_time.strftime("%d %B %Y") %></div>
<div class="text-xs text-slate-600"><%= @ticket.event.start_time.strftime("%H:%M") %></div>
</div>
</div>
<!-- Venue -->
<div class="py-2 border-b border-slate-100">
<span class="text-sm font-medium text-slate-600 block mb-1">Lieu :</span>
<div class="text-sm font-semibold text-slate-900"><%= @ticket.event.venue_name %></div>
<% if @ticket.event.venue_address.present? %>
<div class="text-xs text-slate-600 mt-1"><%= @ticket.event.venue_address %></div>
<% end %>
</div>
</div>
<!-- QR Code Section -->
<div class="bg-slate-50 p-6 text-center border-t border-slate-200">
<h3 class="text-sm font-semibold text-slate-900 mb-4">Code QR du billet</h3>
<div class="inline-block bg-white p-6 rounded-xl shadow-sm border border-slate-200">
<div class="w-52 h-52 flex items-center justify-center qr-code-container">
<%= raw @ticket.generate_qr_svg %>
</div>
</div>
<p class="text-xs text-slate-500 mt-3 font-mono tracking-wider">QR: <%= @ticket.qr_code[0..7] %>...</p>
</div>
<!-- Footer Notice -->
<div class="bg-slate-100 px-6 py-4 text-center border-t border-slate-200">
<div class="space-y-2">
<p class="text-xs text-slate-600">Ce billet est valide pour une seule entrée.</p>
<p class="text-xs text-slate-600">Présentez ce billet à l'entrée du lieu.</p>
<div class="pt-2 border-t border-slate-200">
<p class="text-xs text-slate-500">
Généré le <%= Time.current.strftime('%d %B %Y à %H:%M') %>
</p>
</div>
</div>
</div>
<!-- Action Buttons -->
<div class="p-4 bg-white border-t border-slate-200">
<div class="flex space-x-2">
<%= link_to ticket_path(@ticket),
class: "flex-1 flex items-center justify-center bg-slate-100 hover:bg-slate-200 text-slate-700 py-2.5 px-3 rounded-lg text-sm font-medium transition-colors duration-200" do %>
<svg class="w-4 h-4 mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2">
<path stroke-linecap="round" stroke-linejoin="round" d="M15 12a3 3 0 11-6 0 3 3 0 016 0z"/>
<path stroke-linecap="round" stroke-linejoin="round" d="M2.458 12C3.732 7.943 7.523 5 12 5c4.478 0 8.268 2.943 9.542 7-1.274 4.057-5.064 7-9.542 7-4.477 0-8.268-2.943-9.542-7z"/>
</svg>
Vue détaillée
<% end %>
<% if @ticket.status == 'active' %>
<%= link_to download_ticket_path(@ticket.id),
class: "flex-1 flex items-center justify-center bg-purple-600 hover:bg-purple-700 text-white py-2.5 px-3 rounded-lg text-sm font-medium transition-colors duration-200 shadow-sm hover:shadow-md" do %>
<svg class="w-4 h-4 mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2">
<path stroke-linecap="round" stroke-linejoin="round" d="M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
PDF
<% end %>
<% end %>
</div>
</div>
</div>
<!-- Navigation -->
<div class="text-center mt-6">
<%= link_to dashboard_path, class: "inline-flex items-center text-purple-600 hover:text-purple-800 text-sm font-medium transition-colors duration-200" do %>
<svg class="w-4 h-4 mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2">
<path stroke-linecap="round" stroke-linejoin="round" d="M7 16l-4-4m0 0l4-4m-4 4h18"/>
</svg>
Retour au tableau de bord
<% end %>
</div>
</div>
</div>

View File

@@ -8,7 +8,6 @@
"@hotwired/turbo-rails": "^8.0.13",
"@radix-ui/react-slot": "^1.2.3",
"lucide": "^0.542.0",
"qrcode": "^1.5.4",
"react": "^18.3.1",
"react-dom": "^18.3.1",
},
@@ -145,8 +144,6 @@
"buffer-from": ["buffer-from@1.1.2", "", {}, "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ=="],
"camelcase": ["camelcase@5.3.1", "", {}, "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg=="],
"caniuse-api": ["caniuse-api@3.0.0", "", { "dependencies": { "browserslist": "^4.0.0", "caniuse-lite": "^1.0.0", "lodash.memoize": "^4.1.2", "lodash.uniq": "^4.5.0" } }, "sha512-bsTwuIg/BZZK/vreVTYYbSWoe2F+71P7K5QGEX+pT250DZbfU1MQ5prOKpPR+LL6uWKK3KMwMCAS74QB3Um1uw=="],
"caniuse-lite": ["caniuse-lite@1.0.30001735", "", {}, "sha512-EV/laoX7Wq2J9TQlyIXRxTJqIw4sxfXS4OYgudGxBYRuTv0q7AM6yMEpU/Vo1I94thg9U6EZ2NfZx9GJq83u7w=="],
@@ -203,16 +200,12 @@
"debug": ["debug@4.4.1", "", { "dependencies": { "ms": "^2.1.3" } }, "sha512-KcKCqiftBJcZr++7ykoDIEwSa3XWowTfNPo92BYxjXiyYEVrUQh2aLyhxBCwww+heortUFxEJYcRzosstTEBYQ=="],
"decamelize": ["decamelize@1.2.0", "", {}, "sha512-z2S+W9X73hAUUki+N+9Za2lBlun89zigOyGrsax+KUQ6wKW4ZoWpEYBkGhQjwAjjDCkWxhY0VKEhk8wzY7F5cA=="],
"degenerator": ["degenerator@5.0.1", "", { "dependencies": { "ast-types": "^0.13.4", "escodegen": "^2.1.0", "esprima": "^4.0.1" } }, "sha512-TllpMR/t0M5sqCXfj85i4XaAzxmS5tVA16dqvdkMwGmzI+dXLXnw3J+3Vdv7VKw+ThlTMboK6i9rnZ6Nntj5CQ=="],
"dependency-graph": ["dependency-graph@1.0.0", "", {}, "sha512-cW3gggJ28HZ/LExwxP2B++aiKxhJXMSIt9K48FOXQkm+vuG5gyatXnLsONRJdzO/7VfjDIiaOOa/bs4l464Lwg=="],
"detect-libc": ["detect-libc@2.0.4", "", {}, "sha512-3UDv+G9CsCKO1WKMGw9fwq/SWJYbI0c5Y7LU1AXYoDdbhE2AHQ6N6Nb34sG8Fj7T5APy8qXDCKuuIHd1BR0tVA=="],
"dijkstrajs": ["dijkstrajs@1.0.3", "", {}, "sha512-qiSlmBq9+BCdCA/L46dw8Uy93mloxsPSbwnm5yrKn2vMPiy8KyAskTF6zuV/j5BMsmOGZDPs7KjU+mjb670kfA=="],
"dom-serializer": ["dom-serializer@2.0.0", "", { "dependencies": { "domelementtype": "^2.3.0", "domhandler": "^5.0.2", "entities": "^4.2.0" } }, "sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg=="],
"domelementtype": ["domelementtype@2.3.0", "", {}, "sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw=="],
@@ -257,8 +250,6 @@
"fill-range": ["fill-range@7.1.1", "", { "dependencies": { "to-regex-range": "^5.0.1" } }, "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg=="],
"find-up": ["find-up@4.1.0", "", { "dependencies": { "locate-path": "^5.0.0", "path-exists": "^4.0.0" } }, "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw=="],
"follow-redirects": ["follow-redirects@1.15.11", "", {}, "sha512-deG2P0JfjrTxl50XGCDyfI97ZGVCxIpfKYmfyrQ54n5FO/0gfIES8C/Psl6kWVDolizcaaxZJnTS0QSMxvnsBQ=="],
"fraction.js": ["fraction.js@4.3.7", "", {}, "sha512-ZsDfxO51wGAXREY55a7la9LScWpwv9RxIrYABrlvOFBlH/ShPnrtsXeuUIfXKKOVicNxQ+o8JTbJvjS4M89yew=="],
@@ -341,8 +332,6 @@
"lilconfig": ["lilconfig@3.1.3", "", {}, "sha512-/vlFKAoH5Cgt3Ie+JLhRbwOsCQePABiU3tJ1egGvyQ+33R/vcwM2Zl2QR/LzjsBeItPt3oSVXapn+m4nQDvpzw=="],
"locate-path": ["locate-path@5.0.0", "", { "dependencies": { "p-locate": "^4.1.0" } }, "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g=="],
"lodash": ["lodash@4.17.21", "", {}, "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg=="],
"lodash.memoize": ["lodash.memoize@4.1.2", "", {}, "sha512-t7j+NzmgnQzTAYXcsHYLgimltOV1MXHtlOWf6GjL9Kj8GK5FInw5JotxvbOs+IvV1/Dzo04/fCGfLVs7aXb4Ag=="],
@@ -385,20 +374,12 @@
"nth-check": ["nth-check@2.1.1", "", { "dependencies": { "boolbase": "^1.0.0" } }, "sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w=="],
"p-limit": ["p-limit@2.3.0", "", { "dependencies": { "p-try": "^2.0.0" } }, "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w=="],
"p-locate": ["p-locate@4.1.0", "", { "dependencies": { "p-limit": "^2.2.0" } }, "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A=="],
"p-try": ["p-try@2.2.0", "", {}, "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ=="],
"pac-proxy-agent": ["pac-proxy-agent@7.2.0", "", { "dependencies": { "@tootallnate/quickjs-emscripten": "^0.23.0", "agent-base": "^7.1.2", "debug": "^4.3.4", "get-uri": "^6.0.1", "http-proxy-agent": "^7.0.0", "https-proxy-agent": "^7.0.6", "pac-resolver": "^7.0.1", "socks-proxy-agent": "^8.0.5" } }, "sha512-TEB8ESquiLMc0lV8vcd5Ql/JAKAoyzHFXaStwjkzpOpC5Yv+pIzLfHvjTSdf3vpa2bMiUQrg9i6276yn8666aA=="],
"pac-resolver": ["pac-resolver@7.0.1", "", { "dependencies": { "degenerator": "^5.0.0", "netmask": "^2.0.2" } }, "sha512-5NPgf87AT2STgwa2ntRMr45jTKrYBGkVU36yT0ig/n/GMAa3oPqhZfIQ2kMEimReg0+t9kZViDVZ83qfVUlckg=="],
"pako": ["pako@0.2.9", "", {}, "sha512-NUcwaKxUxWrZLpDG+z/xZaCgQITkA/Dv4V/T6bw7VON6l1Xz/VnrBqrYjZQ12TamKHzITTfOEIYUj48y2KXImA=="],
"path-exists": ["path-exists@4.0.0", "", {}, "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w=="],
"path-parse": ["path-parse@1.0.7", "", {}, "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw=="],
"picocolors": ["picocolors@1.1.1", "", {}, "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA=="],
@@ -421,8 +402,6 @@
"pm2-sysmonit": ["pm2-sysmonit@1.2.8", "", { "dependencies": { "async": "^3.2.0", "debug": "^4.3.1", "pidusage": "^2.0.21", "systeminformation": "^5.7", "tx2": "~1.0.4" } }, "sha512-ACOhlONEXdCTVwKieBIQLSi2tQZ8eKinhcr9JpZSUAL8Qy0ajIgRtsLxG/lwPOW3JEKqPyw/UaHmTWhUzpP4kA=="],
"pngjs": ["pngjs@5.0.0", "", {}, "sha512-40QW5YalBNfQo5yRYmiw7Yz6TKKVr3h6970B2YE+3fQpsWcrbj1PzJgxeJ19DRQjhMbKPIuMY8rFaXc8moolVw=="],
"postcss": ["postcss@8.5.6", "", { "dependencies": { "nanoid": "^3.3.11", "picocolors": "^1.1.1", "source-map-js": "^1.2.1" } }, "sha512-3Ybi1tAuwAP9s0r1UQ2J4n5Y0G05bJkpUIO0/bI9MhwmD70S5aTWbXGBwxHrelT+XM1k6dM0pk+SwNkpTRN7Pg=="],
"postcss-calc": ["postcss-calc@10.1.1", "", { "dependencies": { "postcss-selector-parser": "^7.0.0", "postcss-value-parser": "^4.2.0" }, "peerDependencies": { "postcss": "^8.4.38" } }, "sha512-NYEsLHh8DgG/PRH2+G9BTuUdtf9ViS+vdoQ0YA5OQdGsfN4ztiwtDWNtBl9EKeqNMFnIu8IKZ0cLxEQ5r5KVMw=="],
@@ -505,8 +484,6 @@
"proxy-from-env": ["proxy-from-env@1.1.0", "", {}, "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg=="],
"qrcode": ["qrcode@1.5.4", "", { "dependencies": { "dijkstrajs": "^1.0.1", "pngjs": "^5.0.0", "yargs": "^15.3.1" }, "bin": { "qrcode": "bin/qrcode" } }, "sha512-1ca71Zgiu6ORjHqFBDpnSMTR2ReToX4l1Au1VFLyVeBTFavzQnv5JxMFr3ukHVKpSrSA2MCk0lNJSykjUfz7Zg=="],
"react": ["react@18.3.1", "", { "dependencies": { "loose-envify": "^1.1.0" } }, "sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ=="],
"react-dom": ["react-dom@18.3.1", "", { "dependencies": { "loose-envify": "^1.1.0", "scheduler": "^0.23.2" }, "peerDependencies": { "react": "^18.3.1" } }, "sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw=="],
@@ -521,8 +498,6 @@
"require-in-the-middle": ["require-in-the-middle@5.2.0", "", { "dependencies": { "debug": "^4.1.1", "module-details-from-path": "^1.0.3", "resolve": "^1.22.1" } }, "sha512-efCx3b+0Z69/LGJmm9Yvi4cqEdxnoGnxYxGxBghkkTTFeXRtTCmmhO0AnAfHz59k957uTSuy8WaHqOs8wbYUWg=="],
"require-main-filename": ["require-main-filename@2.0.0", "", {}, "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg=="],
"resolve": ["resolve@1.22.10", "", { "dependencies": { "is-core-module": "^2.16.0", "path-parse": "^1.0.7", "supports-preserve-symlinks-flag": "^1.0.0" }, "bin": "bin/resolve" }, "sha512-NPRy+/ncIMeDlTAsuqwKIiferiawhefFJtkNSW0qZJEqMEb+qBt/77B/jGeeek+F0uOeN05CDa6HXbbIgtVX4w=="],
"run-series": ["run-series@1.1.9", "", {}, "sha512-Arc4hUN896vjkqCYrUXquBFtRZdv1PfLbTYP71efP6butxyQ0kWpiNJyAgsxscmQg1cqvHY32/UCBzXedTpU2g=="],
@@ -537,8 +512,6 @@
"semver": ["semver@7.7.2", "", { "bin": "bin/semver.js" }, "sha512-RF0Fw+rO5AMf9MAyaRXI4AV0Ulj5lMHqVxxdSgiVbixSCXoEmmX/jk0CuJw4+3SqroYO9VoUh+HcuJivvtJemA=="],
"set-blocking": ["set-blocking@2.0.0", "", {}, "sha512-KiKBS8AnWGEyLzofFfmvKwpdPzqiy16LvQfK3yv/fVH7Bj13/wl3JSR1J+rfgRE9q7xUJK4qvgS8raSOeLUehw=="],
"shimmer": ["shimmer@1.2.1", "", {}, "sha512-sQTKC1Re/rM6XyFM6fIAGHRPVGvyXfgzIDvzoq608vM+jeyVD0Tu1E6Np0Kc2zAIFWIj963V2800iF/9LPieQw=="],
"signal-exit": ["signal-exit@3.0.7", "", {}, "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ=="],
@@ -603,8 +576,6 @@
"vizion": ["vizion@2.2.1", "", { "dependencies": { "async": "^2.6.3", "git-node-fs": "^1.0.0", "ini": "^1.3.5", "js-git": "^0.7.8" } }, "sha512-sfAcO2yeSU0CSPFI/DmZp3FsFE9T+8913nv1xWBOyzODv13fwkn6Vl7HqxGpkr9F608M+8SuFId3s+BlZqfXww=="],
"which-module": ["which-module@2.0.1", "", {}, "sha512-iBdZ57RDvnOR9AGBhML2vFZf7h8vmBjhoaZqODJBFWHVtKkDmKuHai3cx5PgVMrX5YDNp27AofYbAwctSS+vhQ=="],
"wrap-ansi": ["wrap-ansi@7.0.0", "", { "dependencies": { "ansi-styles": "^4.0.0", "string-width": "^4.1.0", "strip-ansi": "^6.0.0" } }, "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q=="],
"ws": ["ws@7.5.10", "", { "peerDependencies": { "bufferutil": "^4.0.1", "utf-8-validate": "^5.0.2" }, "optionalPeers": ["bufferutil", "utf-8-validate"] }, "sha512-+dbF1tHwZpXcbOJdVOkzLDxZP1ailvSxM6ZweXTegylPny803bFhA+vqBYw4s31NSAk4S2Qz+AKXK9a4wkdjcQ=="],
@@ -649,8 +620,6 @@
"pm2-sysmonit/pidusage": ["pidusage@2.0.21", "", { "dependencies": { "safe-buffer": "^5.2.1" } }, "sha512-cv3xAQos+pugVX+BfXpHsbyz/dLzX+lr44zNMsYiGxUw+kV5sgQCIcLd1z+0vq+KyC7dJ+/ts2PsfgWfSC3WXA=="],
"qrcode/yargs": ["yargs@15.4.1", "", { "dependencies": { "cliui": "^6.0.0", "decamelize": "^1.2.0", "find-up": "^4.1.0", "get-caller-file": "^2.0.1", "require-directory": "^2.1.1", "require-main-filename": "^2.0.0", "set-blocking": "^2.0.0", "string-width": "^4.2.0", "which-module": "^2.0.0", "y18n": "^4.0.0", "yargs-parser": "^18.1.2" } }, "sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A=="],
"readdirp/picomatch": ["picomatch@2.3.1", "", {}, "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA=="],
"svgo/commander": ["commander@11.1.0", "", {}, "sha512-yPVavfyCcRhmorC7rWlkHn15b4wDVgVmBA7kV4QVBsF7kv/9TKJAbAXVTxvTnwP8HHKjRCJDClKbciiYS7p0DQ=="],
@@ -665,16 +634,8 @@
"csso/css-tree/mdn-data": ["mdn-data@2.0.28", "", {}, "sha512-aylIc7Z9y4yzHYAJNuESG3hfhC+0Ibp/MAMiaOZgNv4pmEdFyfZhhhny4MNiAfWdBQ1RQ2mfDWmM1x8SvGyp8g=="],
"qrcode/yargs/cliui": ["cliui@6.0.0", "", { "dependencies": { "string-width": "^4.2.0", "strip-ansi": "^6.0.0", "wrap-ansi": "^6.2.0" } }, "sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ=="],
"qrcode/yargs/y18n": ["y18n@4.0.3", "", {}, "sha512-JKhqTOwSrqNA1NY5lSztJ1GrBiUodLMmIZuLiDaMRJ+itFd+ABVE8XBjOvIWL+rSqNDC74LCSFmlb/U4UZ4hJQ=="],
"qrcode/yargs/yargs-parser": ["yargs-parser@18.1.3", "", { "dependencies": { "camelcase": "^5.0.0", "decamelize": "^1.2.0" } }, "sha512-o50j0JeToy/4K6OZcaQmW6lyXXKhq7csREXcDwk2omFPJEwUNOVtJKvmDr9EI1fAJZUyZcRF7kxGBWmRXudrCQ=="],
"@pm2/agent/semver/lru-cache/yallist": ["yallist@4.0.0", "", {}, "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A=="],
"@pm2/io/semver/lru-cache/yallist": ["yallist@4.0.0", "", {}, "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A=="],
"qrcode/yargs/cliui/wrap-ansi": ["wrap-ansi@6.2.0", "", { "dependencies": { "ansi-styles": "^4.0.0", "string-width": "^4.1.0", "strip-ansi": "^6.0.0" } }, "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA=="],
}
}

View File

@@ -1,23 +0,0 @@
# Disable view annotations for mailer templates to prevent HTML comments
# from breaking email formatting in development mode
if Rails.env.development?
Rails.application.configure do
# Override the annotation setting for ActionMailer specifically
config.to_prepare do
ActionMailer::Base.prepend(Module.new do
def mail(headers = {}, &block)
# Temporarily disable view annotations during email rendering
original_setting = ActionView::Base.annotate_rendered_view_with_filenames
ActionView::Base.annotate_rendered_view_with_filenames = false
result = super(headers, &block)
# Restore original setting
ActionView::Base.annotate_rendered_view_with_filenames = original_setting
result
end
end)
end
end
end

View File

@@ -1,21 +0,0 @@
# Schedule event reminder notifications
Rails.application.config.after_initialize do
# Only schedule in production or when SCHEDULE_REMINDERS is set
if Rails.env.production? || ENV["SCHEDULE_REMINDERS"] == "true"
# Schedule the reminder scheduler to run daily at 9 AM
begin
# Use a simple cron-like approach with ActiveJob
# This will be handled by solid_queue in production
EventReminderSchedulerJob.set(wait_until: next_run_time).perform_later
rescue StandardError => e
Rails.logger.warn "Could not schedule event reminders: #{e.message}"
end
end
end
def next_run_time
# Schedule for 9 AM today, or 9 AM tomorrow if it's already past 9 AM
target_time = Time.current.beginning_of_day + 9.hours
target_time += 1.day if Time.current > target_time
target_time
end

View File

@@ -31,10 +31,6 @@ Rails.application.routes.draw do
confirmation: "auth/confirmations" # Custom controller for confirmations
}
# === Onboarding ===
get "onboarding", to: "onboarding#index", as: "onboarding"
post "onboarding", to: "onboarding#complete", as: "complete_onboarding"
# === Pages ===
get "dashboard", to: "pages#dashboard", as: "dashboard"
@@ -46,10 +42,9 @@ Rails.application.routes.draw do
get "orders/new/events/:slug.:id", to: "orders#new", as: "event_order_new"
post "orders/create/events/:slug.:id", to: "orders#create", as: "event_order_create"
resources :orders, only: [ :index, :show ] do
resources :orders, only: [ :show ] do
member do
get :checkout
get :invoice
post :retry_payment
post :increment_payment_attempt
end
@@ -58,15 +53,16 @@ Rails.application.routes.draw do
get "orders/payments/success", to: "orders#payment_success", as: "order_payment_success"
get "orders/payments/cancel", to: "orders#payment_cancel", as: "order_payment_cancel"
# Legacy routes - redirect to order system
get "events/:slug.:id/tickets/checkout", to: "tickets#checkout", as: "ticket_checkout"
post "events/:slug.:id/tickets/retry", to: "tickets#retry_payment", as: "ticket_retry_payment"
get "payments/success", to: "tickets#payment_success", as: "payment_success"
get "payments/cancel", to: "tickets#payment_cancel", as: "payment_cancel"
# legacy routes
get "payments/success", to: "tickets#payment_success", as: "payment_success"
get "payments/cancel", to: "tickets#payment_cancel", as: "payment_cancel"
# === Tickets ===
get "tickets/:qr_code", to: "tickets#show", as: "ticket"
get "tickets/:qr_code/download", to: "tickets#download", as: "ticket_download"
get "tickets/checkout/events/:slug.:id", to: "tickets#checkout", as: "ticket_checkout"
post "tickets/retry/events/:slug.:id", to: "tickets#retry_payment", as: "ticket_retry_payment"
get "tickets/:ticket_id", to: "tickets#show", as: "ticket"
get "tickets/:ticket_id/view", to: "tickets#ticket_view", as: "ticket_view"
get "tickets/:ticket_id/download", to: "tickets#download_ticket", as: "download_ticket"
# === Promoter Routes ===
namespace :promoter do

View File

@@ -1,5 +0,0 @@
class AddOnboardingToUsers < ActiveRecord::Migration[8.0]
def change
add_column :users, :onboarding_completed, :boolean, default: false, null: false
end
end

3
db/schema.rb generated
View File

@@ -10,7 +10,7 @@
#
# It's strongly recommended that you check this file into your version control system.
ActiveRecord::Schema[8.0].define(version: 2025_09_08_092220) do
ActiveRecord::Schema[8.0].define(version: 2025_08_23_171354) do
create_table "events", charset: "utf8mb4", collation: "utf8mb4_uca1400_ai_ci", force: :cascade do |t|
t.string "name", null: false
t.string "slug", null: false
@@ -94,7 +94,6 @@ ActiveRecord::Schema[8.0].define(version: 2025_09_08_092220) do
t.string "stripe_customer_id"
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
t.boolean "onboarding_completed", default: false, null: false
t.index ["email"], name: "index_users_on_email", unique: true
t.index ["reset_password_token"], name: "index_users_on_reset_password_token", unique: true
end

View File

@@ -53,7 +53,6 @@ services:
mailhog:
image: corpusops/mailhog:v1.0.1
restart: unless-stopped
# environment:
# - "mh_auth_file=/opt/mailhog/passwd.conf"
volumes:

File diff suppressed because it is too large Load Diff

View File

@@ -1,162 +0,0 @@
# Email Notifications System
This document describes the email notifications system implemented for ApéroNight.
## Overview
The email notifications system provides two main types of notifications:
1. **Purchase Confirmation Emails** - Sent when orders are completed
2. **Event Reminder Emails** - Sent at scheduled intervals before events
## Features
### Purchase Confirmation Emails
- **Trigger**: Automatically sent when an order is marked as paid
- **Content**: Order details, ticket information, PDF attachments for each ticket
- **Template**: Supports both single tickets and multi-ticket orders
- **Languages**: French (can be extended)
### Event Reminder Emails
- **Schedule**: 7 days before, 1 day before, and day of event
- **Content**: Event details, user's ticket information, venue information
- **Recipients**: Only users with active tickets for the event
- **Smart Content**: Different messaging based on time until event
## Technical Implementation
### Mailer Classes
#### TicketMailer
- `purchase_confirmation_order(order)` - For complete orders with multiple tickets
- `purchase_confirmation(ticket)` - For individual tickets
- `event_reminder(user, event, days_before)` - For event reminders
### Background Jobs
#### EventReminderJob
- Sends reminder emails to all users with active tickets for a specific event
- Parameters: `event_id`, `days_before`
- Error handling: Logs failures but continues processing other users
#### EventReminderSchedulerJob
- Runs daily to schedule reminder emails
- Automatically finds events starting in 7 days, 1 day, or same day
- Only processes published events
- Configurable via environment variables
### Email Templates
Templates are available in both HTML and text formats:
- `app/views/ticket_mailer/purchase_confirmation.html.erb`
- `app/views/ticket_mailer/purchase_confirmation.text.erb`
- `app/views/ticket_mailer/event_reminder.html.erb`
- `app/views/ticket_mailer/event_reminder.text.erb`
### Configuration
#### Environment Variables
- `MAILER_FROM_EMAIL` - From address for emails (default: no-reply@aperonight.fr)
- `SMTP_*` - SMTP configuration for production
- `SCHEDULE_REMINDERS` - Enable automatic reminder scheduling in non-production
#### Development Setup
- Uses localhost:1025 for development (MailCatcher recommended)
- Email delivery is configured but won't raise errors in development
## Usage
### Manual Testing
```ruby
# Test purchase confirmation
order = Order.last
TicketMailer.purchase_confirmation_order(order).deliver_now
# Test event reminder
user = User.first
event = Event.published.first
TicketMailer.event_reminder(user, event, 7).deliver_now
# Test scheduler job
EventReminderSchedulerJob.perform_now
```
### Integration in Code
Purchase confirmation emails are automatically sent when orders are marked as paid:
```ruby
order.mark_as_paid! # Automatically sends confirmation email
```
Event reminders are automatically scheduled via the initializer, but can be manually triggered:
```ruby
# Schedule reminders for a specific event
EventReminderJob.perform_later(event.id, 7) # 7 days before
```
## Deployment Notes
### Production Configuration
1. Configure SMTP settings via environment variables
2. Set `MAILER_FROM_EMAIL` to your domain
3. Ensure `SCHEDULE_REMINDERS=true` to enable automatic reminders
4. Configure solid_queue for background job processing
### Monitoring
- Check logs for email delivery failures
- Monitor job queue for stuck reminder jobs
- Verify SMTP configuration is working
### Customization
- Email templates can be customized in `app/views/ticket_mailer/`
- Add new reminder intervals by modifying `EventReminderSchedulerJob`
- Internationalization can be added using Rails I18n
## File Structure
```
app/
├── jobs/
│ ├── event_reminder_job.rb
│ └── event_reminder_scheduler_job.rb
├── mailers/
│ ├── application_mailer.rb
│ └── ticket_mailer.rb
└── views/
└── ticket_mailer/
├── purchase_confirmation.html.erb
├── purchase_confirmation.text.erb
├── event_reminder.html.erb
└── event_reminder.text.erb
config/
├── environments/
│ ├── development.rb (SMTP localhost:1025)
│ └── production.rb (ENV-based SMTP)
└── initializers/
└── event_reminder_scheduler.rb
test/
├── jobs/
│ ├── event_reminder_job_test.rb
│ └── event_reminder_scheduler_job_test.rb
├── mailers/
│ └── ticket_mailer_test.rb
└── integration/
└── email_notifications_integration_test.rb
```
## Security Considerations
- No sensitive information in email templates
- User data is properly escaped in templates
- QR codes contain only necessary ticket verification data
- Email addresses are validated through Devise

View File

@@ -1,275 +0,0 @@
# Ticket Download Security Implementation
## Overview
This document describes how to implement secure unique identifiers for ticket PDF downloads to enhance security and prevent unauthorized access to user tickets.
## Problem Statement
Currently, the ticket download functionality uses the QR code directly as an identifier in URLs. This approach presents several security risks:
1. **Predictability**: QR codes may follow predictable patterns
2. **Information Disclosure**: QR codes might reveal internal system information
3. **Brute Force Vulnerability**: Attackers can enumerate valid tickets
4. **Lack of Revocability**: Cannot invalidate download links without affecting the QR code
## Solution
Implement a separate, cryptographically secure unique identifier specifically for PDF downloads.
## Implementation Steps
### 1. Database Migration
Create a migration to add the new column:
```ruby
# db/migrate/xxx_add_pdf_download_token_to_tickets.rb
class AddPdfDownloadTokenToTickets < ActiveRecord::Migration[7.0]
def change
add_column :tickets, :pdf_download_token, :string, limit: 50
add_column :tickets, :pdf_download_token_expires_at, :datetime
add_index :tickets, :pdf_download_token, unique: true
end
end
```
### 2. Model Implementation
Update the Ticket model to generate secure tokens:
```ruby
# app/models/ticket.rb
class Ticket < ApplicationRecord
before_create :generate_pdf_download_token
# Generate a secure token for PDF downloads
def generate_pdf_download_token
self.pdf_download_token = SecureRandom.urlsafe_base64(32)
self.pdf_download_token_expires_at = 24.hours.from_now
end
# Check if the download token is still valid
def pdf_download_token_valid?
pdf_download_token.present? &&
pdf_download_token_expires_at.present? &&
pdf_download_token_expires_at > Time.current
end
# Regenerate token (useful for security or when token expires)
def regenerate_pdf_download_token
generate_pdf_download_token
save!
end
# Ensure tokens are generated for existing records
def ensure_pdf_download_token
if pdf_download_token.blank?
generate_pdf_download_token
save!
end
end
end
```
### 3. Controller Updates
Update the TicketsController to use the new token system:
```ruby
# app/controllers/tickets_controller.rb
class TicketsController < ApplicationController
before_action :authenticate_user!
def show
@ticket = Ticket.joins(order: :user)
.includes(:event, :ticket_type, order: :user)
.find_by(tickets: { qr_code: params[:qr_code] })
if @ticket.nil?
redirect_to dashboard_path, alert: "Billet non trouvé"
return
end
@event = @ticket.event
@order = @ticket.order
end
def download
# Find ticket by PDF download token instead of QR code
@ticket = Ticket.find_by(pdf_download_token: params[:pdf_download_token])
# Check if ticket exists
if @ticket.nil?
redirect_to dashboard_path, alert: "Lien de téléchargement invalide ou expiré"
return
end
# Verify token validity
unless @ticket.pdf_download_token_valid?
redirect_to dashboard_path, alert: "Le lien de téléchargement a expiré"
return
end
# Verify ownership
unless @ticket.order.user == current_user
redirect_to dashboard_path, alert: "Vous n'avez pas l'autorisation d'accéder à ce billet"
return
end
# Generate and send PDF
pdf_content = @ticket.to_pdf
# Optionally regenerate token to make it single-use
# @ticket.regenerate_pdf_download_token
send_data pdf_content,
filename: "ticket_#{@ticket.id}_#{@ticket.event.name.parameterize}.pdf",
type: "application/pdf",
disposition: "attachment"
rescue => e
Rails.logger.error "Error generating ticket PDF: #{e.message}"
redirect_to dashboard_path, alert: "Erreur lors de la génération du billet"
end
end
```
### 4. Route Configuration
Update routes to use the new token-based system:
```ruby
# config/routes.rb
Rails.application.routes.draw do
# Existing routes...
# Update ticket download route
get "tickets/:pdf_download_token/download", to: "tickets#download", as: "ticket_download"
# Keep existing show route for QR code functionality
get "tickets/:qr_code", to: "tickets#show", as: "ticket"
end
```
### 5. View Updates
Update views to use the new download URL:
```erb
<!-- In app/views/tickets/show.html.erb -->
<%= link_to ticket_download_path(@ticket.pdf_download_token),
class: "flex-1 bg-gradient-to-r from-purple-600 to-indigo-600 hover:from-purple-700 hover:to-indigo-700 text-white font-medium py-3 px-6 rounded-xl shadow-sm transition-all duration-200 focus:outline-none focus:ring-2 focus:ring-purple-500 focus:ring-offset-2 transform hover:-translate-y-0.5 text-center" do %>
<svg class="w-4 h-4 inline-block mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
Télécharger le PDF
<% end %>
```
### 6. Background Job for Token Management
Create a job to clean up expired tokens periodically:
```ruby
# app/jobs/cleanup_expired_ticket_tokens_job.rb
class CleanupExpiredTicketTokensJob < ApplicationJob
queue_as :default
def perform
# Clear expired tokens to free up database space
Ticket.where("pdf_download_token_expires_at < ?", 1.week.ago)
.update_all(pdf_download_token: nil, pdf_download_token_expires_at: nil)
end
end
```
Schedule this job to run regularly:
```ruby
# config/schedule.rb (if using whenever gem)
every 1.day, at: '4:30 am' do
rake "tickets:cleanup_expired_tokens"
end
```
## Security Benefits
1. **Unpredictability**: Tokens are cryptographically secure and random
2. **Separation of Concerns**: QR codes for physical entry, tokens for digital downloads
3. **Revocability**: Tokens can be regenerated without affecting QR codes
4. **Expirability**: Time-limited access prevents long-term exposure
5. **Ownership Verification**: Additional checks ensure only ticket owners can download
6. **Audit Trail**: Token usage can be logged for security monitoring
## Additional Security Considerations
### Rate Limiting
Implement rate limiting to prevent abuse:
```ruby
# In ApplicationController or specific controller
before_action :rate_limit_downloads, only: [:download]
def rate_limit_downloads
if Rails.cache.read("download_attempts_#{current_user.id}")&.to_i > 10
render json: { error: "Too many download attempts" }, status: :too_many_requests
else
Rails.cache.write("download_attempts_#{current_user.id}",
(Rails.cache.read("download_attempts_#{current_user.id}") || 0) + 1,
expires_in: 1.hour)
end
end
```
### Logging
Add logging for security monitoring:
```ruby
# In TicketsController#download
Rails.logger.info "Ticket PDF download attempted - User: #{current_user.id}, Ticket: #{@ticket.id}, Token: #{params[:pdf_download_token]}"
```
## Migration Process
1. Run the database migration
2. Update existing tickets with tokens:
```ruby
# In rails console or a rake task
Ticket.find_each(&:ensure_pdf_download_token)
```
3. Deploy code changes
4. Update any external references to use the new system
5. Monitor for issues and adjust expiration times as needed
## Testing
Ensure comprehensive testing of the new functionality:
```ruby
# spec/controllers/tickets_controller_spec.rb
RSpec.describe TicketsController, type: :controller do
describe "GET #download" do
it "downloads PDF for valid token" do
# Test implementation
end
it "rejects expired tokens" do
# Test implementation
end
it "rejects invalid tokens" do
# Test implementation
end
it "verifies ticket ownership" do
# Test implementation
end
end
end
```
## Conclusion
This implementation provides a robust security framework for ticket PDF downloads while maintaining usability. The separation of QR codes (for physical entry) and download tokens (for digital access) follows security best practices and provides multiple layers of protection against unauthorized access.

740
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@@ -11,7 +11,7 @@
"@hotwired/turbo-rails": "^8.0.13",
"@radix-ui/react-slot": "^1.2.3",
"lucide": "^0.542.0",
"qrcode": "^1.5.4",
"puppeteer": "^24.19.0",
"react": "^18.3.1",
"react-dom": "^18.3.1"
},
@@ -32,5 +32,21 @@
"tailwind-merge": "^3.3.1",
"tailwindcss": "^4.1.4",
"tailwindcss-animate": "^1.0.7"
}
},
"version": "1.0.0",
"description": "![Aperonight Screenshot](app/assets/images/screenshot-homepage.png)",
"main": "ecosystem.config.js",
"directories": {
"doc": "docs",
"lib": "lib",
"test": "test"
},
"repository": {
"type": "git",
"url": "ssh://git@gitea.cyanet.fr:2222/kbe/aperonight.git"
},
"keywords": [],
"author": "",
"license": "ISC",
"type": "commonjs"
}

View File

@@ -1,57 +0,0 @@
require "test_helper"
class ApplicationControllerOnboardingTest < ActionDispatch::IntegrationTest
setup do
@user_without_onboarding = users(:one)
@user_without_onboarding.update!(onboarding_completed: false)
@user_with_onboarding = users(:two)
@user_with_onboarding.update!(onboarding_completed: true, first_name: "John", last_name: "Doe")
end
test "should redirect incomplete users to onboarding from dashboard" do
sign_in @user_without_onboarding
get dashboard_path
assert_redirected_to onboarding_path
end
test "should allow complete users to access dashboard" do
sign_in @user_with_onboarding
get dashboard_path
assert_response :success
end
test "should redirect incomplete users to onboarding from events" do
sign_in @user_without_onboarding
get events_path
assert_redirected_to onboarding_path
end
test "should allow complete users to access events" do
sign_in @user_with_onboarding
get events_path
assert_response :success
end
test "should not redirect from home page when not signed in" do
get root_path
assert_response :success
end
test "should redirect signed in incomplete users from home to onboarding" do
sign_in @user_without_onboarding
get root_path
assert_redirected_to dashboard_path # Home redirects to dashboard for signed in users
end
test "should not interfere with devise controllers" do
get new_user_session_path
assert_response :success
end
test "should not redirect when already on onboarding page" do
sign_in @user_without_onboarding
get onboarding_path
assert_response :success
end
end

View File

@@ -1,86 +0,0 @@
require "test_helper"
class OnboardingControllerTest < ActionDispatch::IntegrationTest
setup do
@user_without_onboarding = users(:one)
@user_without_onboarding.update!(onboarding_completed: false)
@user_with_onboarding = users(:two)
@user_with_onboarding.update!(onboarding_completed: true, first_name: "John", last_name: "Doe")
end
test "should redirect to onboarding when user not signed in" do
get onboarding_path
assert_redirected_to new_user_session_path
end
test "should show onboarding page for incomplete user" do
sign_in @user_without_onboarding
get onboarding_path
assert_response :success
assert_select "h1", /Bienvenue sur.*!/
assert_select "form"
end
test "should redirect completed user to dashboard" do
sign_in @user_with_onboarding
get onboarding_path
assert_redirected_to dashboard_path
end
test "should complete onboarding with valid data" do
sign_in @user_without_onboarding
assert_not @user_without_onboarding.onboarding_completed?
post complete_onboarding_path, params: {
user: {
first_name: "Jane",
last_name: "Smith"
}
}
assert_redirected_to dashboard_path
follow_redirect!
assert_select ".notification", /Bienvenue sur AperoNight/
@user_without_onboarding.reload
assert @user_without_onboarding.onboarding_completed?
assert_equal "Jane", @user_without_onboarding.first_name
assert_equal "Smith", @user_without_onboarding.last_name
end
test "should not complete onboarding without required fields" do
sign_in @user_without_onboarding
post complete_onboarding_path, params: {
user: {
first_name: "",
last_name: "Smith"
}
}
assert_response :success
assert_select ".notification", /Veuillez remplir tous les champs requis/
@user_without_onboarding.reload
assert_not @user_without_onboarding.onboarding_completed?
end
test "should not complete onboarding without last name" do
sign_in @user_without_onboarding
post complete_onboarding_path, params: {
user: {
first_name: "Jane",
last_name: ""
}
}
assert_response :success
assert_select ".notification", /Veuillez remplir tous les champs requis/
@user_without_onboarding.reload
assert_not @user_without_onboarding.onboarding_completed?
end
end

View File

@@ -1,23 +0,0 @@
require "test_helper"
class OrdersControllerInvoiceTest < ActionDispatch::IntegrationTest
def setup
@user = users(:one)
@event = events(:concert_event)
@order = orders(:paid_order)
sign_in @user
end
test "should get invoice for paid order" do
get invoice_order_url(@order)
assert_response :success
assert_select "h1", "Facture"
end
test "should redirect to order page for unpaid order" do
draft_order = orders(:draft_order)
get invoice_order_url(draft_order)
assert_redirected_to order_url(draft_order)
assert_equal "La facture n'est disponible qu'après le paiement de la commande", flash[:alert]
end
end

View File

@@ -5,10 +5,7 @@ class OrdersControllerTest < ActionDispatch::IntegrationTest
@user = User.create!(
email: "test@example.com",
password: "password123",
password_confirmation: "password123",
onboarding_completed: true,
first_name: "Test",
last_name: "User"
password_confirmation: "password123"
)
@event = Event.create!(
@@ -125,7 +122,7 @@ class OrdersControllerTest < ActionDispatch::IntegrationTest
assert_equal "draft", new_order.status
assert_equal @user, new_order.user
assert_equal @event, new_order.event
assert_equal @ticket_type.price_cents + 100, new_order.total_amount_cents # includes 1€ service fee
assert_equal @ticket_type.price_cents, new_order.total_amount_cents
assert_redirected_to checkout_order_path(new_order)
assert_equal new_order.id, session[:draft_order_id]

View File

@@ -5,11 +5,9 @@ one:
encrypted_password: <%= Devise::Encryptor.digest(User, 'password123') %>
last_name: Trump
first_name: Donald
onboarding_completed: true
two:
email: user2@example.com
encrypted_password: <%= Devise::Encryptor.digest(User, 'password123') %>
last_name: Obama
first_name: Barack
onboarding_completed: true

View File

@@ -1,101 +0,0 @@
require "test_helper"
class EmailNotificationsIntegrationTest < ActionDispatch::IntegrationTest
include ActiveJob::TestHelper
def setup
@user = User.create!(
email: "test@example.com",
password: "password123",
first_name: "Test",
last_name: "User"
)
@event = Event.create!(
name: "Test Event",
slug: "test-event",
description: "A test event for integration testing",
state: :published,
venue_name: "Test Venue",
venue_address: "123 Test Street",
latitude: 40.7128,
longitude: -74.0060,
start_time: 1.week.from_now,
end_time: 1.week.from_now + 4.hours,
user: @user
)
@ticket_type = TicketType.create!(
name: "General Admission",
description: "General admission ticket",
price_cents: 2500,
quantity: 100,
sale_start_at: 1.day.ago,
sale_end_at: 1.day.from_now,
event: @event
)
@order = Order.create!(
user: @user,
event: @event,
status: "draft",
total_amount_cents: 2500,
payment_attempts: 0
)
@ticket = Ticket.create!(
order: @order,
ticket_type: @ticket_type,
first_name: "Test",
last_name: "User",
price_cents: 2500,
status: "draft"
)
end
test "sends purchase confirmation email when order is marked as paid" do
# Mock PDF generation to avoid QR code issues
@ticket.stubs(:to_pdf).returns("fake_pdf_content")
assert_emails 1 do
@order.mark_as_paid!
end
assert_equal "paid", @order.status
assert_equal "active", @ticket.reload.status
end
test "event reminder email can be sent to users with active tickets" do
# Setup: mark order as paid and activate tickets
@ticket.stubs(:to_pdf).returns("fake_pdf_content")
@order.mark_as_paid!
# Clear any emails from the setup
ActionMailer::Base.deliveries.clear
assert_emails 1 do
TicketMailer.event_reminder(@user, @event, 7).deliver_now
end
email = ActionMailer::Base.deliveries.last
assert_equal [ @user.email ], email.to
assert_equal "Rappel : #{@event.name} dans une semaine", email.subject
end
test "event reminder job schedules emails for users with tickets" do
# Setup: mark order as paid and activate tickets
@ticket.stubs(:to_pdf).returns("fake_pdf_content")
@order.mark_as_paid!
# Clear any emails from the setup
ActionMailer::Base.deliveries.clear
# Perform the job
EventReminderJob.perform_now(@event.id, 7)
assert_equal 1, ActionMailer::Base.deliveries.size
email = ActionMailer::Base.deliveries.last
assert_equal [ @user.email ], email.to
assert_match "une semaine", email.subject
end
end

View File

@@ -1,31 +0,0 @@
require "test_helper"
class EventReminderJobTest < ActiveJob::TestCase
def setup
@event = events(:concert_event)
@user = users(:one)
@ticket = tickets(:one)
end
test "performs event reminder job for users with tickets" do
# Mock the mailer to avoid actual email sending in tests
TicketMailer.expects(:event_reminder).with(@user, @event, 7).returns(stub(deliver_now: true))
EventReminderJob.perform_now(@event.id, 7)
end
test "handles missing event gracefully" do
assert_raises(ActiveRecord::RecordNotFound) do
EventReminderJob.perform_now(999999, 7)
end
end
test "logs error when mailer fails" do
# Mock a failing mailer
TicketMailer.stubs(:event_reminder).raises(StandardError.new("Test error"))
Rails.logger.expects(:error).with(regexp_matches(/Failed to send event reminder/))
EventReminderJob.perform_now(@event.id, 7)
end
end

View File

@@ -1,50 +0,0 @@
require "test_helper"
class EventReminderSchedulerJobTest < ActiveJob::TestCase
def setup
@event = events(:concert_event)
end
test "schedules weekly reminders for events starting in 7 days" do
# Set event to start in exactly 7 days
@event.update(start_time: 7.days.from_now.beginning_of_day + 10.hours)
assert_enqueued_with(job: EventReminderJob, args: [ @event.id, 7 ]) do
EventReminderSchedulerJob.perform_now
end
end
test "schedules daily reminders for events starting tomorrow" do
# Set event to start tomorrow
@event.update(start_time: 1.day.from_now.beginning_of_day + 20.hours)
assert_enqueued_with(job: EventReminderJob, args: [ @event.id, 1 ]) do
EventReminderSchedulerJob.perform_now
end
end
test "schedules day-of reminders for events starting today" do
# Set event to start today
@event.update(start_time: Time.current.beginning_of_day + 21.hours)
assert_enqueued_with(job: EventReminderJob, args: [ @event.id, 0 ]) do
EventReminderSchedulerJob.perform_now
end
end
test "does not schedule reminders for draft events" do
@event.update(state: :draft, start_time: 7.days.from_now.beginning_of_day + 10.hours)
assert_no_enqueued_jobs(only: EventReminderJob) do
EventReminderSchedulerJob.perform_now
end
end
test "does not schedule reminders for cancelled events" do
@event.update(state: :canceled, start_time: 7.days.from_now.beginning_of_day + 10.hours)
assert_no_enqueued_jobs(only: EventReminderJob) do
EventReminderSchedulerJob.perform_now
end
end
end

View File

@@ -1,104 +0,0 @@
require "test_helper"
class TicketMailerTest < ActionMailer::TestCase
def setup
@user = users(:one)
@event = events(:concert_event)
@ticket_type = ticket_types(:standard)
@order = orders(:paid_order)
@ticket = tickets(:one)
end
test "purchase confirmation order email" do
# Mock PDF generation for all tickets
@order.tickets.each do |ticket|
ticket.stubs(:to_pdf).returns("fake_pdf_data")
end
email = TicketMailer.purchase_confirmation_order(@order)
assert_emails 1 do
email.deliver_now
end
assert_equal [ "no-reply@aperonight.fr" ], email.from
assert_equal [ @user.email ], email.to
assert_equal "Confirmation d'achat - #{@event.name}", email.subject
assert_match @event.name, email.body.to_s
assert_match @user.email.split("@").first, email.body.to_s
end
test "purchase confirmation single ticket email" do
# Mock PDF generation
@ticket.stubs(:to_pdf).returns("fake_pdf_data")
email = TicketMailer.purchase_confirmation(@ticket)
assert_emails 1 do
email.deliver_now
end
assert_equal [ "no-reply@aperonight.fr" ], email.from
assert_equal [ @ticket.user.email ], email.to
assert_equal "Confirmation d'achat - #{@ticket.event.name}", email.subject
assert_match @ticket.event.name, email.body.to_s
assert_match @ticket.user.email.split("@").first, email.body.to_s
end
test "event reminder email one week before" do
# Ensure the user has active tickets for the event by using the existing fixtures
# The 'one' ticket fixture is already linked to the 'paid_order' and 'concert_event'
email = TicketMailer.event_reminder(@user, @event, 7)
# Only test delivery if the user has tickets (the method returns early if not)
if email
assert_emails 1 do
email.deliver_now
end
assert_equal [ "no-reply@aperonight.fr" ], email.from
assert_equal [ @user.email ], email.to
assert_equal "Rappel : #{@event.name} dans une semaine", email.subject
assert_match "une semaine", email.body.to_s
assert_match @event.name, email.body.to_s
else
# If no email is sent, that's expected behavior when user has no active tickets
assert_no_emails do
TicketMailer.event_reminder(@user, @event, 7)
end
end
end
test "event reminder email one day before" do
email = TicketMailer.event_reminder(@user, @event, 1)
assert_emails 1 do
email.deliver_now
end
assert_equal "Rappel : #{@event.name} demain", email.subject
assert_match "demain", email.body.to_s
end
test "event reminder email day of event" do
email = TicketMailer.event_reminder(@user, @event, 0)
assert_emails 1 do
email.deliver_now
end
assert_equal "C'est aujourd'hui : #{@event.name}", email.subject
assert_match "aujourd'hui", email.body.to_s
end
test "event reminder email custom days" do
email = TicketMailer.event_reminder(@user, @event, 3)
assert_emails 1 do
email.deliver_now
end
assert_equal "Rappel : #{@event.name} dans 3 jours", email.subject
assert_match "3 jours", email.body.to_s
end
end

View File

@@ -1,38 +0,0 @@
require "test_helper"
class OrderEmailTest < ActiveSupport::TestCase
def setup
@order = orders(:draft_order)
end
test "sends purchase confirmation email when order is marked as paid" do
# Mock the mailer to capture the call
TicketMailer.expects(:purchase_confirmation_order).with(@order).returns(stub(deliver_now: true))
@order.mark_as_paid!
assert_equal "paid", @order.status
end
test "activates all tickets when order is marked as paid" do
@order.tickets.update_all(status: "reserved")
# Mock the mailer to avoid actual email sending
TicketMailer.stubs(:purchase_confirmation_order).returns(stub(deliver_now: true))
@order.mark_as_paid!
assert @order.tickets.all? { |ticket| ticket.status == "active" }
end
test "email sending failure does not prevent order completion" do
# Mock mailer to raise an error
TicketMailer.stubs(:purchase_confirmation_order).raises(StandardError.new("Email error"))
# Should not raise error - email failure is logged but doesn't fail the payment
@order.mark_as_paid!
# Order should still be marked as paid even if email fails
assert_equal "paid", @order.reload.status
end
end

View File

@@ -469,7 +469,7 @@ class OrderTest < ActiveSupport::TestCase
assert_equal "active", ticket2.status
end
test "calculate_total! should sum ticket prices plus 1€ service fee" do
test "calculate_total! should sum ticket prices" do
order = Order.create!(
user: @user, event: @event, total_amount_cents: 0,
status: "draft", payment_attempts: 0
@@ -506,7 +506,7 @@ class OrderTest < ActiveSupport::TestCase
order.calculate_total!
order.reload
assert_equal 3100, order.total_amount_cents # 2 tickets * 1500 cents + 100 cents (1€ fee)
assert_equal 3000, order.total_amount_cents # 2 tickets * 1500 cents
end
# === Stripe Integration Tests (Mock) ===

View File

@@ -63,33 +63,4 @@ class UserTest < ActiveSupport::TestCase
refute user.valid?, "User with last_name longer than 12 chars should be invalid"
assert_not_nil user.errors[:last_name], "No validation error for too long last_name"
end
# Test onboarding functionality
test "new users should need onboarding by default" do
user = User.new(email: "test@example.com", password: "password123")
assert user.needs_onboarding?, "New user should need onboarding"
assert_not user.onboarding_completed?, "New user should not have completed onboarding"
end
test "should complete onboarding" do
user = users(:one)
user.update!(onboarding_completed: false)
assert user.needs_onboarding?, "User should need onboarding initially"
user.complete_onboarding!
assert_not user.needs_onboarding?, "User should not need onboarding after completion"
assert user.onboarding_completed?, "User should have completed onboarding"
end
test "needs_onboarding? should return correct value" do
user = users(:one)
user.update!(onboarding_completed: false)
assert user.needs_onboarding?, "User with false onboarding_completed should need onboarding"
user.update!(onboarding_completed: true)
assert_not user.needs_onboarding?, "User with true onboarding_completed should not need onboarding"
end
end

View File

@@ -151,7 +151,7 @@ class StripeInvoiceServiceTest < ActiveSupport::TestCase
mock_invoice.stubs(:finalize_invoice).returns(mock_invoice)
mock_invoice.expects(:pay)
Stripe::Invoice.expects(:create).returns(mock_invoice)
Stripe::InvoiceItem.expects(:create).twice # Once for tickets, once for service fee
Stripe::InvoiceItem.expects(:create).once
result = @service.create_post_payment_invoice
assert_not_nil result
@@ -173,7 +173,7 @@ class StripeInvoiceServiceTest < ActiveSupport::TestCase
mock_invoice.stubs(:finalize_invoice).returns(mock_invoice)
mock_invoice.expects(:pay)
Stripe::Invoice.expects(:create).returns(mock_invoice)
Stripe::InvoiceItem.expects(:create).twice # Once for tickets, once for service fee
Stripe::InvoiceItem.expects(:create).once
result = @service.create_post_payment_invoice
assert_not_nil result
@@ -196,7 +196,7 @@ class StripeInvoiceServiceTest < ActiveSupport::TestCase
mock_customer.stubs(:id).returns("cus_test123")
Stripe::Customer.expects(:create).returns(mock_customer)
expected_ticket_line_item = {
expected_line_item = {
customer: "cus_test123",
invoice: "in_test123",
amount: @ticket_type.price_cents * 2, # 2 tickets
@@ -210,25 +210,12 @@ class StripeInvoiceServiceTest < ActiveSupport::TestCase
}
}
expected_service_fee_line_item = {
customer: "cus_test123",
invoice: "in_test123",
amount: 100,
currency: "eur",
description: "Frais de service - Frais de traitement de la commande",
metadata: {
item_type: "service_fee",
amount_cents: 100
}
}
mock_invoice = mock("invoice")
mock_invoice.stubs(:id).returns("in_test123")
mock_invoice.stubs(:finalize_invoice).returns(mock_invoice)
mock_invoice.expects(:pay)
Stripe::Invoice.expects(:create).returns(mock_invoice)
Stripe::InvoiceItem.expects(:create).with(expected_ticket_line_item)
Stripe::InvoiceItem.expects(:create).with(expected_service_fee_line_item)
Stripe::InvoiceItem.expects(:create).with(expected_line_item)
result = @service.create_post_payment_invoice
assert_not_nil result
@@ -261,7 +248,7 @@ class StripeInvoiceServiceTest < ActiveSupport::TestCase
mock_invoice.expects(:pay)
Stripe::Invoice.expects(:create).with(expected_invoice_data).returns(mock_invoice)
Stripe::InvoiceItem.expects(:create).twice # Once for tickets, once for service fee
Stripe::InvoiceItem.expects(:create).once
result = @service.create_post_payment_invoice
assert_not_nil result
@@ -300,7 +287,7 @@ class StripeInvoiceServiceTest < ActiveSupport::TestCase
})
Stripe::Invoice.expects(:create).returns(mock_invoice)
Stripe::InvoiceItem.expects(:create).twice # Once for tickets, once for service fee
Stripe::InvoiceItem.expects(:create).once
mock_invoice.expects(:finalize_invoice).returns(mock_finalized_invoice)
result = @service.create_post_payment_invoice

View File

@@ -1,283 +0,0 @@
require "test_helper"
class TicketPdfGeneratorTest < ActiveSupport::TestCase
def setup
# Stub QR code generation to avoid dependency issues
mock_qrcode = mock("qrcode")
mock_qrcode.stubs(:modules).returns([])
RQRCode::QRCode.stubs(:new).returns(mock_qrcode)
@user = User.create!(
email: "test@example.com",
password: "password123",
password_confirmation: "password123"
)
@event = Event.create!(
name: "Test Event",
slug: "test-event",
description: "A valid description for the test event that is long enough",
latitude: 48.8566,
longitude: 2.3522,
venue_name: "Test Venue",
venue_address: "123 Test Street",
user: @user,
start_time: 1.week.from_now,
end_time: 1.week.from_now + 3.hours,
state: :published
)
@ticket_type = TicketType.create!(
name: "General Admission",
description: "General admission tickets with full access to the event",
price_cents: 2500,
quantity: 100,
sale_start_at: Time.current,
sale_end_at: @event.start_time - 1.hour,
requires_id: false,
event: @event
)
@order = Order.create!(
user: @user,
event: @event,
status: "paid",
total_amount_cents: 2500
)
@ticket = Ticket.create!(
order: @order,
ticket_type: @ticket_type,
status: "active",
first_name: "John",
last_name: "Doe",
qr_code: "test-qr-code-123"
)
end
# === Initialization Tests ===
test "should initialize with ticket" do
generator = TicketPdfGenerator.new(@ticket)
assert_equal @ticket, generator.ticket
end
# === PDF Generation Tests ===
test "should generate PDF for valid ticket" do
generator = TicketPdfGenerator.new(@ticket)
pdf_string = generator.generate
assert_not_nil pdf_string
assert_kind_of String, pdf_string
assert pdf_string.length > 0
# Check if it starts with PDF header
assert pdf_string.start_with?("%PDF")
end
test "should include event name in PDF" do
generator = TicketPdfGenerator.new(@ticket)
# Test that PDF generates successfully
pdf_string = generator.generate
assert_not_nil pdf_string
assert pdf_string.start_with?("%PDF")
assert pdf_string.length > 1000, "PDF should be substantial in size"
end
test "should include ticket type information in PDF" do
generator = TicketPdfGenerator.new(@ticket)
pdf_string = generator.generate
# Basic check that PDF was generated - actual content validation
# would require parsing the PDF which is complex
assert_not_nil pdf_string
assert pdf_string.length > 0
end
test "should include price information in PDF" do
generator = TicketPdfGenerator.new(@ticket)
pdf_string = generator.generate
assert_not_nil pdf_string
assert pdf_string.length > 0
end
test "should include venue information in PDF" do
generator = TicketPdfGenerator.new(@ticket)
pdf_string = generator.generate
assert_not_nil pdf_string
assert pdf_string.length > 0
end
test "should include QR code in PDF" do
generator = TicketPdfGenerator.new(@ticket)
# Just test that PDF generates successfully
pdf_string = generator.generate
assert_not_nil pdf_string
assert pdf_string.length > 0
assert pdf_string.start_with?("%PDF")
end
# === Error Handling Tests ===
test "should raise error when QR code is blank" do
# Create ticket with blank QR code (skip validations)
ticket_with_blank_qr = Ticket.new(
order: @order,
ticket_type: @ticket_type,
status: "active",
first_name: "John",
last_name: "Doe",
price_cents: 2500,
qr_code: ""
)
ticket_with_blank_qr.save(validate: false)
generator = TicketPdfGenerator.new(ticket_with_blank_qr)
error = assert_raises(RuntimeError) do
generator.generate
end
assert_equal "Ticket QR code is missing", error.message
end
test "should raise error when QR code is nil" do
# Create ticket with nil QR code (skip validations)
ticket_with_nil_qr = Ticket.new(
order: @order,
ticket_type: @ticket_type,
status: "active",
first_name: "John",
last_name: "Doe",
price_cents: 2500,
qr_code: nil
)
ticket_with_nil_qr.save(validate: false)
generator = TicketPdfGenerator.new(ticket_with_nil_qr)
error = assert_raises(RuntimeError) do
generator.generate
end
assert_equal "Ticket QR code is missing", error.message
end
test "should handle missing event gracefully in QR data" do
# Create ticket with minimal data but valid QR code
orphaned_ticket = Ticket.new(
order: @order,
ticket_type: @ticket_type,
status: "active",
first_name: "John",
last_name: "Doe",
price_cents: 2500,
qr_code: "test-qr-code-orphaned"
)
orphaned_ticket.save(validate: false)
generator = TicketPdfGenerator.new(orphaned_ticket)
# Should still generate PDF
pdf_string = generator.generate
assert_not_nil pdf_string
assert pdf_string.length > 0
assert pdf_string.start_with?("%PDF")
end
# === QR Code Data Tests ===
test "should generate correct QR code data" do
generator = TicketPdfGenerator.new(@ticket)
# Just test that PDF generates successfully with QR data
pdf_string = generator.generate
assert_not_nil pdf_string
assert pdf_string.start_with?("%PDF")
end
test "should compact QR code data removing nils" do
# Test with a ticket that has unique QR code
ticket_with_minimal_data = Ticket.new(
order: @order,
ticket_type: @ticket_type,
status: "active",
first_name: "Jane",
last_name: "Smith",
price_cents: 2500,
qr_code: "test-qr-minimal-data"
)
ticket_with_minimal_data.save(validate: false)
generator = TicketPdfGenerator.new(ticket_with_minimal_data)
# Should generate PDF successfully
pdf_string = generator.generate
assert_not_nil pdf_string
assert pdf_string.start_with?("%PDF")
end
# === Price Display Tests ===
test "should format price correctly in euros" do
# Test different price formats
@ticket.update!(price_cents: 1050) # €10.50
generator = TicketPdfGenerator.new(@ticket)
pdf_string = generator.generate
assert_not_nil pdf_string
assert_equal 10.5, @ticket.price_euros
end
test "should handle low price" do
@ticket_type.update!(price_cents: 1)
@ticket.update!(price_cents: 1)
generator = TicketPdfGenerator.new(@ticket)
pdf_string = generator.generate
assert_not_nil pdf_string
assert_equal 0.01, @ticket.price_euros
end
# === Date Formatting Tests ===
test "should format event date correctly" do
specific_time = Time.parse("2024-12-25 19:30:00")
@event.update!(start_time: specific_time)
generator = TicketPdfGenerator.new(@ticket)
pdf_string = generator.generate
# Just verify PDF generates - date formatting is handled by strftime
assert_not_nil pdf_string
assert pdf_string.length > 0
end
# === Integration Tests ===
test "should generate valid PDF with all required elements" do
generator = TicketPdfGenerator.new(@ticket)
pdf_string = generator.generate
# Basic PDF structure validation
assert_not_nil pdf_string
assert pdf_string.start_with?("%PDF")
assert pdf_string.end_with?("%%EOF\n")
assert pdf_string.length > 1000, "PDF should be substantial in size"
end
test "should be callable from ticket model" do
# Test the integration with the Ticket model's to_pdf method
pdf_string = @ticket.to_pdf
assert_not_nil pdf_string
assert pdf_string.start_with?("%PDF")
end
end

View File

@@ -17,18 +17,6 @@ module ActiveSupport
fixtures :all
# Add more helper methods to be used by all tests here...
# Helper to create users with completed onboarding by default for tests
def create_test_user(attributes = {})
User.create!({
email: "test#{rand(10000)}@example.com",
password: "password123",
password_confirmation: "password123",
first_name: "Test",
last_name: "User",
onboarding_completed: true
}.merge(attributes))
end
end
end

717
yarn.lock

File diff suppressed because it is too large Load Diff