feat: page configuration paiement contrat (CB + SEPA) + email automatique
Apres signature du contrat, le webhook envoie un email avec lien
vers /process/contrat/{id}/setup-payment
Page setup-payment :
- Resume montant mensuel HT
- Choix CB (Stripe Checkout avec setup_future_usage) ou SEPA
- Formulaire IBAN Stripe Elements avec mandat SEPA
- Confirmation SEPA via endpoint POST /confirm
- Page succes apres paiement
Routes :
- /process/contrat/{id}/setup-payment : page choix CB/SEPA
- /process/contrat/{id}/setup-payment/confirm : confirmation SEPA
- /process/contrat/{id}/payment-success : page succes
Email contrat_setup_payment : lien vers la page de configuration,
detail montant, mention 1er paiement CB/SEPA obligatoire
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -8,9 +8,11 @@ use App\Service\MailerService;
|
|||||||
use Doctrine\ORM\EntityManagerInterface;
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
|
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
|
||||||
use Symfony\Component\DependencyInjection\Attribute\Autowire;
|
use Symfony\Component\DependencyInjection\Attribute\Autowire;
|
||||||
|
use Symfony\Component\HttpFoundation\JsonResponse;
|
||||||
use Symfony\Component\HttpFoundation\Request;
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
use Symfony\Component\HttpFoundation\Response;
|
use Symfony\Component\HttpFoundation\Response;
|
||||||
use Symfony\Component\Routing\Attribute\Route;
|
use Symfony\Component\Routing\Attribute\Route;
|
||||||
|
use Symfony\Component\Routing\Generator\UrlGeneratorInterface;
|
||||||
use Twig\Environment;
|
use Twig\Environment;
|
||||||
|
|
||||||
class ContratProcessController extends AbstractController
|
class ContratProcessController extends AbstractController
|
||||||
@@ -118,6 +120,160 @@ class ContratProcessController extends AbstractController
|
|||||||
throw $this->createNotFoundException('Lien de signature introuvable.');
|
throw $this->createNotFoundException('Lien de signature introuvable.');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Page de configuration du paiement (CB ou SEPA).
|
||||||
|
*/
|
||||||
|
#[Route('/process/contrat/{id}/setup-payment', name: 'app_contrat_setup_payment', requirements: ['id' => '\d+'])]
|
||||||
|
public function setupPayment(
|
||||||
|
int $id,
|
||||||
|
Request $request,
|
||||||
|
#[Autowire(env: 'STRIPE_SK')] string $stripeSk = '',
|
||||||
|
#[Autowire(env: 'STRIPE_PK')] string $stripePk = '',
|
||||||
|
): Response {
|
||||||
|
$contrat = $this->em->getRepository(Contrat::class)->find($id);
|
||||||
|
if (null === $contrat || Contrat::STATE_SIGNED !== $contrat->getState()) {
|
||||||
|
throw $this->createNotFoundException('Contrat introuvable.');
|
||||||
|
}
|
||||||
|
|
||||||
|
$session = $request->getSession();
|
||||||
|
if (!$session->get('contrat_verified_'.$contrat->getId(), false)) {
|
||||||
|
return $this->redirectToRoute('app_contrat_verify', ['id' => $id]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$customer = $contrat->getCustomer();
|
||||||
|
if (null === $customer) {
|
||||||
|
throw $this->createNotFoundException('Client introuvable.');
|
||||||
|
}
|
||||||
|
|
||||||
|
// @codeCoverageIgnoreStart
|
||||||
|
\Stripe\Stripe::setApiKey($stripeSk);
|
||||||
|
|
||||||
|
$stripeCustomerId = $customer->getStripeCustomerId();
|
||||||
|
if (null === $stripeCustomerId) {
|
||||||
|
$stripeCustomer = \Stripe\Customer::create([
|
||||||
|
'email' => $customer->getEmail() ?? $contrat->getEmail(),
|
||||||
|
'name' => $contrat->getRaisonSociale(),
|
||||||
|
]);
|
||||||
|
$stripeCustomerId = $stripeCustomer->id;
|
||||||
|
$customer->setStripeCustomerId($stripeCustomerId);
|
||||||
|
$this->em->flush();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Creer un SetupIntent pour SEPA
|
||||||
|
$setupIntent = \Stripe\SetupIntent::create([
|
||||||
|
'customer' => $stripeCustomerId,
|
||||||
|
'payment_method_types' => ['sepa_debit'],
|
||||||
|
'metadata' => ['contrat_id' => (string) $contrat->getId()],
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Creer aussi un Checkout Session pour CB (premier paiement)
|
||||||
|
$totalCents = (int) round($contrat->getTotalHt() * 100);
|
||||||
|
|
||||||
|
$cbCheckoutUrl = null;
|
||||||
|
if ($totalCents > 0) {
|
||||||
|
$checkoutSession = \Stripe\Checkout\Session::create([
|
||||||
|
'mode' => 'payment',
|
||||||
|
'payment_method_types' => ['card'],
|
||||||
|
'customer' => $stripeCustomerId,
|
||||||
|
'line_items' => [[
|
||||||
|
'price_data' => [
|
||||||
|
'currency' => 'eur',
|
||||||
|
'unit_amount' => $totalCents,
|
||||||
|
'product_data' => [
|
||||||
|
'name' => 'Premier paiement - '.$contrat->getReference(),
|
||||||
|
],
|
||||||
|
],
|
||||||
|
'quantity' => 1,
|
||||||
|
]],
|
||||||
|
'payment_intent_data' => [
|
||||||
|
'setup_future_usage' => 'off_session',
|
||||||
|
'metadata' => [
|
||||||
|
'contrat_id' => (string) $contrat->getId(),
|
||||||
|
'first_payment' => '1',
|
||||||
|
],
|
||||||
|
],
|
||||||
|
'success_url' => $this->generateUrl('app_contrat_payment_success', ['id' => $id], UrlGeneratorInterface::ABSOLUTE_URL),
|
||||||
|
'cancel_url' => $this->generateUrl('app_contrat_setup_payment', ['id' => $id], UrlGeneratorInterface::ABSOLUTE_URL),
|
||||||
|
]);
|
||||||
|
$cbCheckoutUrl = $checkoutSession->url;
|
||||||
|
}
|
||||||
|
// @codeCoverageIgnoreEnd
|
||||||
|
|
||||||
|
return $this->render('contrat/setup_payment.html.twig', [
|
||||||
|
'contrat' => $contrat,
|
||||||
|
'customer' => $customer,
|
||||||
|
'clientSecret' => $setupIntent->client_secret,
|
||||||
|
'stripePk' => $stripePk,
|
||||||
|
'cbCheckoutUrl' => $cbCheckoutUrl,
|
||||||
|
'totalHt' => $contrat->getTotalHt(),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Confirmation SEPA pour contrat.
|
||||||
|
*/
|
||||||
|
#[Route('/process/contrat/{id}/setup-payment/confirm', name: 'app_contrat_setup_payment_confirm', requirements: ['id' => '\d+'], methods: ['POST'])]
|
||||||
|
public function setupPaymentConfirm(
|
||||||
|
int $id,
|
||||||
|
Request $request,
|
||||||
|
#[Autowire(env: 'STRIPE_SK')] string $stripeSk = '',
|
||||||
|
): Response {
|
||||||
|
$contrat = $this->em->getRepository(Contrat::class)->find($id);
|
||||||
|
if (null === $contrat) {
|
||||||
|
return new JsonResponse(['error' => 'Contrat introuvable'], Response::HTTP_NOT_FOUND);
|
||||||
|
}
|
||||||
|
|
||||||
|
$session = $request->getSession();
|
||||||
|
if (!$session->get('contrat_verified_'.$contrat->getId(), false)) {
|
||||||
|
return new JsonResponse(['error' => 'Non autorise'], Response::HTTP_FORBIDDEN);
|
||||||
|
}
|
||||||
|
|
||||||
|
$data = json_decode($request->getContent(), true);
|
||||||
|
$paymentMethodId = $data['payment_method'] ?? null;
|
||||||
|
|
||||||
|
if (null === $paymentMethodId) {
|
||||||
|
return new JsonResponse(['error' => 'payment_method manquant'], Response::HTTP_BAD_REQUEST);
|
||||||
|
}
|
||||||
|
|
||||||
|
$customer = $contrat->getCustomer();
|
||||||
|
if (null === $customer) {
|
||||||
|
return new JsonResponse(['error' => 'Client introuvable'], Response::HTTP_NOT_FOUND);
|
||||||
|
}
|
||||||
|
|
||||||
|
// @codeCoverageIgnoreStart
|
||||||
|
try {
|
||||||
|
\Stripe\Stripe::setApiKey($stripeSk);
|
||||||
|
|
||||||
|
$stripeCustomerId = $customer->getStripeCustomerId();
|
||||||
|
$pm = \Stripe\PaymentMethod::retrieve($paymentMethodId);
|
||||||
|
$pm->attach(['customer' => $stripeCustomerId]);
|
||||||
|
\Stripe\Customer::update($stripeCustomerId, [
|
||||||
|
'invoice_settings' => ['default_payment_method' => $paymentMethodId],
|
||||||
|
]);
|
||||||
|
} catch (\Throwable $e) {
|
||||||
|
return new JsonResponse(['error' => $e->getMessage()], Response::HTTP_INTERNAL_SERVER_ERROR);
|
||||||
|
}
|
||||||
|
// @codeCoverageIgnoreEnd
|
||||||
|
|
||||||
|
return new JsonResponse(['status' => 'ok']);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Page succes apres premier paiement CB.
|
||||||
|
*/
|
||||||
|
#[Route('/process/contrat/{id}/payment-success', name: 'app_contrat_payment_success', requirements: ['id' => '\d+'])]
|
||||||
|
public function paymentSuccess(int $id): Response
|
||||||
|
{
|
||||||
|
$contrat = $this->em->getRepository(Contrat::class)->find($id);
|
||||||
|
if (null === $contrat) {
|
||||||
|
throw $this->createNotFoundException('Contrat introuvable.');
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->render('contrat/payment_success.html.twig', [
|
||||||
|
'contrat' => $contrat,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
private function sendVerificationCode(Contrat $contrat, object $session, int $id): void
|
private function sendVerificationCode(Contrat $contrat, object $session, int $id): void
|
||||||
{
|
{
|
||||||
$code = str_pad((string) random_int(0, 999999), 6, '0', \STR_PAD_LEFT);
|
$code = str_pad((string) random_int(0, 999999), 6, '0', \STR_PAD_LEFT);
|
||||||
|
|||||||
@@ -311,6 +311,30 @@ class WebhookDocuSealController extends AbstractController
|
|||||||
// silencieux
|
// silencieux
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Mail client : configuration du paiement
|
||||||
|
if (null !== $customer) {
|
||||||
|
try {
|
||||||
|
$paymentUrl = $this->generateUrl('app_contrat_setup_payment', [
|
||||||
|
'id' => $contrat->getId(),
|
||||||
|
], \Symfony\Component\Routing\Generator\UrlGeneratorInterface::ABSOLUTE_URL);
|
||||||
|
|
||||||
|
$mailer->sendEmail(
|
||||||
|
$contrat->getEmail(),
|
||||||
|
'Configurez votre paiement - '.$contrat->getReference(),
|
||||||
|
$twig->render('emails/contrat_setup_payment.html.twig', [
|
||||||
|
'contrat' => $contrat,
|
||||||
|
'customer' => $customer,
|
||||||
|
'paymentUrl' => $paymentUrl,
|
||||||
|
]),
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
false,
|
||||||
|
);
|
||||||
|
} catch (\Throwable) {
|
||||||
|
// silencieux
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return new JsonResponse(['status' => 'ok', 'event' => 'contrat_signed', 'reference' => $contrat->getReference(), 'customer_created' => null !== $customer]);
|
return new JsonResponse(['status' => 'ok', 'event' => 'contrat_signed', 'reference' => $contrat->getReference(), 'customer_created' => null !== $customer]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
36
templates/contrat/payment_success.html.twig
Normal file
36
templates/contrat/payment_success.html.twig
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
{% extends 'base.html.twig' %}
|
||||||
|
|
||||||
|
{% block title %}Paiement configure - {{ contrat.reference }} - Association E-Cosplay{% endblock %}
|
||||||
|
|
||||||
|
{% block body %}
|
||||||
|
<div class="min-h-screen flex items-center justify-center p-4" style="background: linear-gradient(135deg, #f5f5f0 0%, #e8e8e0 100%);">
|
||||||
|
<div class="glass-heavy w-full max-w-lg overflow-hidden">
|
||||||
|
<div class="glass-dark text-white px-8 py-6">
|
||||||
|
<div class="flex items-center gap-3">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" class="h-10 w-10 text-green-400" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z" />
|
||||||
|
</svg>
|
||||||
|
<div>
|
||||||
|
<h1 class="text-lg font-bold uppercase tracking-widest">Paiement configure</h1>
|
||||||
|
<p class="text-xs text-white/60">{{ contrat.reference }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="p-8 text-center">
|
||||||
|
<p class="text-sm text-gray-600 mb-4">
|
||||||
|
Votre mode de paiement a ete configure avec succes. Vos services seront actifs prochainement.
|
||||||
|
</p>
|
||||||
|
<div class="glass p-4 mb-4 text-left">
|
||||||
|
<p class="text-xs text-gray-500"><strong>Contrat :</strong> {{ contrat.reference }}</p>
|
||||||
|
<p class="text-xs text-gray-500"><strong>Montant :</strong> {{ contrat.totalHt|number_format(2, ',', ' ') }} € HT / mois</p>
|
||||||
|
</div>
|
||||||
|
<p class="text-xs text-gray-400">
|
||||||
|
Vous recevrez un email de confirmation a chaque prelevement. Votre espace client est accessible sur <a href="https://client.e-cosplay.fr" class="font-bold" style="color: #fabf04;">client.e-cosplay.fr</a>
|
||||||
|
</p>
|
||||||
|
<p class="text-xs text-gray-400 mt-4">
|
||||||
|
Pour toute question : <a href="mailto:client@e-cosplay.fr" class="font-bold" style="color: #fabf04;">client@e-cosplay.fr</a>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endblock %}
|
||||||
140
templates/contrat/setup_payment.html.twig
Normal file
140
templates/contrat/setup_payment.html.twig
Normal file
@@ -0,0 +1,140 @@
|
|||||||
|
{% extends 'base.html.twig' %}
|
||||||
|
|
||||||
|
{% block title %}Configuration paiement - {{ contrat.reference }} - Association E-Cosplay{% endblock %}
|
||||||
|
|
||||||
|
{% block body %}
|
||||||
|
<div class="min-h-screen flex items-center justify-center p-4" style="background: linear-gradient(135deg, #f5f5f0 0%, #e8e8e0 100%);">
|
||||||
|
<div class="glass-heavy w-full max-w-2xl overflow-hidden">
|
||||||
|
<div class="glass-dark text-white px-8 py-6">
|
||||||
|
<div class="flex items-center gap-3">
|
||||||
|
<img src="/logo.jpg" alt="E-Cosplay" class="h-10 w-auto">
|
||||||
|
<div>
|
||||||
|
<h1 class="text-lg font-bold uppercase tracking-widest">Configuration du paiement</h1>
|
||||||
|
<p class="text-xs text-white/60">{{ contrat.reference }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="p-8">
|
||||||
|
<p class="text-sm text-gray-600 mb-6">Choisissez votre mode de paiement pour activer vos services.</p>
|
||||||
|
|
||||||
|
{# Resume #}
|
||||||
|
<div class="glass p-4 mb-6 text-center">
|
||||||
|
<p class="text-[9px] font-bold uppercase tracking-wider text-gray-400">Montant mensuel</p>
|
||||||
|
<p class="text-3xl font-bold mt-1" style="color: #fabf04;">{{ totalHt|number_format(2, ',', ' ') }} € <span class="text-sm text-gray-500">HT / mois</span></p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{# Choix methode #}
|
||||||
|
<h2 class="text-sm font-bold uppercase tracking-wider mb-4">Comment souhaitez-vous payer ?</h2>
|
||||||
|
<div class="grid grid-cols-1 md:grid-cols-2 gap-4 mb-6">
|
||||||
|
{# Carte bancaire #}
|
||||||
|
<div class="glass p-5 text-center hover:bg-white/80 transition-all">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" class="h-8 w-8 mx-auto mb-2 text-purple-600" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" 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>
|
||||||
|
<p class="text-xs font-bold uppercase tracking-wider mb-2">Carte bancaire</p>
|
||||||
|
<p class="text-[10px] text-gray-500 mb-3">Payez votre premier mois immediatement par carte bancaire via Stripe.</p>
|
||||||
|
{% if cbCheckoutUrl %}
|
||||||
|
<a href="{{ cbCheckoutUrl }}"
|
||||||
|
class="inline-block px-4 py-2 bg-purple-600 text-white font-bold uppercase text-[10px] tracking-wider hover:bg-purple-700 transition-all">
|
||||||
|
Payer {{ totalHt|number_format(2, ',', ' ') }} € par CB
|
||||||
|
</a>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{# SEPA #}
|
||||||
|
<div class="glass p-5 text-center hover:bg-white/80 transition-all" style="border: 2px solid #fabf04;">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" class="h-8 w-8 mx-auto mb-2" style="color: #fabf04;" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" d="M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15" />
|
||||||
|
</svg>
|
||||||
|
<p class="text-xs font-bold uppercase tracking-wider mb-2">Prelevement SEPA</p>
|
||||||
|
<span class="inline-block px-2 py-0.5 text-[9px] font-bold uppercase tracking-wider mb-2" style="background: #fabf04; color: #111;">Recommande</span>
|
||||||
|
<p class="text-[10px] text-gray-500 mb-3">Renseignez votre IBAN une seule fois. Les prelevements seront automatiques chaque mois.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{# Formulaire SEPA #}
|
||||||
|
<div class="glass p-5 mb-6">
|
||||||
|
<h3 class="text-sm font-bold uppercase tracking-wider mb-3">Configurer le prelevement SEPA</h3>
|
||||||
|
<form id="sepa-form">
|
||||||
|
<div class="mb-3">
|
||||||
|
<label for="account-name" class="block text-[9px] font-bold uppercase tracking-wider text-gray-400 mb-1">Titulaire du compte</label>
|
||||||
|
<input type="text" id="account-name" required value="{{ contrat.raisonSociale }}" class="input-glass w-full px-4 py-3 text-sm font-bold">
|
||||||
|
</div>
|
||||||
|
<div class="mb-3">
|
||||||
|
<label for="account-email" class="block text-[9px] font-bold uppercase tracking-wider text-gray-400 mb-1">Email</label>
|
||||||
|
<input type="email" id="account-email" required value="{{ contrat.email }}" class="input-glass w-full px-4 py-3 text-sm font-bold">
|
||||||
|
</div>
|
||||||
|
<div class="mb-3">
|
||||||
|
<label class="block text-[9px] font-bold uppercase tracking-wider text-gray-400 mb-1">IBAN</label>
|
||||||
|
<div id="iban-element" class="input-glass w-full px-4 py-3"></div>
|
||||||
|
<div id="iban-errors" class="text-red-500 text-xs mt-1 hidden"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="glass p-3 mb-4 text-xs text-gray-500 leading-relaxed">
|
||||||
|
<p class="font-bold text-[9px] uppercase tracking-wider text-gray-400 mb-1">Mandat SEPA</p>
|
||||||
|
<p>En fournissant vos informations de paiement, vous autorisez Association E-Cosplay et Stripe a debiter votre compte conformement aux instructions.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="form-error" class="mb-3 p-3 bg-red-500/20 text-red-700 font-bold text-xs hidden"></div>
|
||||||
|
|
||||||
|
<button type="submit" id="submit-btn" class="w-full btn-gold px-4 py-3 font-bold uppercase text-xs tracking-wider text-gray-900 disabled:opacity-50">
|
||||||
|
<span id="btn-text">Autoriser le prelevement SEPA</span>
|
||||||
|
<span id="btn-loading" class="hidden">Traitement en cours...</span>
|
||||||
|
</button>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p class="text-center text-xs text-gray-400">Pour toute question : <a href="mailto:client@e-cosplay.fr" class="font-bold" style="color: #fabf04;">client@e-cosplay.fr</a></p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script src="https://js.stripe.com/v3/" nonce="{{ csp_nonce('script') }}"></script>
|
||||||
|
<script nonce="{{ csp_nonce('script') }}">
|
||||||
|
(function() {
|
||||||
|
var stripe = Stripe('{{ stripePk }}');
|
||||||
|
var elements = stripe.elements();
|
||||||
|
var style = { base: { color: '#111827', fontSize: '14px', fontFamily: 'Arial, sans-serif', '::placeholder': { color: '#9ca3af' } }, invalid: { color: '#dc2626' } };
|
||||||
|
var iban = elements.create('iban', { style: style, supportedCountries: ['SEPA'] });
|
||||||
|
iban.mount('#iban-element');
|
||||||
|
|
||||||
|
var errorEl = document.getElementById('iban-errors');
|
||||||
|
iban.on('change', function(event) {
|
||||||
|
if (event.error) { errorEl.textContent = event.error.message; errorEl.classList.remove('hidden'); }
|
||||||
|
else { errorEl.classList.add('hidden'); }
|
||||||
|
});
|
||||||
|
|
||||||
|
var form = document.getElementById('sepa-form');
|
||||||
|
var submitBtn = document.getElementById('submit-btn');
|
||||||
|
var btnText = document.getElementById('btn-text');
|
||||||
|
var btnLoading = document.getElementById('btn-loading');
|
||||||
|
var formError = document.getElementById('form-error');
|
||||||
|
|
||||||
|
form.addEventListener('submit', function(e) {
|
||||||
|
e.preventDefault();
|
||||||
|
submitBtn.disabled = true;
|
||||||
|
btnText.classList.add('hidden');
|
||||||
|
btnLoading.classList.remove('hidden');
|
||||||
|
formError.classList.add('hidden');
|
||||||
|
|
||||||
|
stripe.confirmSepaDebitSetup('{{ clientSecret }}', {
|
||||||
|
payment_method: { sepa_debit: iban, billing_details: { name: document.getElementById('account-name').value, email: document.getElementById('account-email').value } }
|
||||||
|
}).then(function(result) {
|
||||||
|
if (result.error) {
|
||||||
|
formError.textContent = result.error.message; formError.classList.remove('hidden');
|
||||||
|
submitBtn.disabled = false; btnText.classList.remove('hidden'); btnLoading.classList.add('hidden');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
fetch('{{ path('app_contrat_setup_payment_confirm', {id: contrat.id}) }}', {
|
||||||
|
method: 'POST', headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({ payment_method: result.setupIntent.payment_method })
|
||||||
|
}).then(function(res) { return res.json(); }).then(function(data) {
|
||||||
|
if (data.status === 'ok') { window.location.href = '{{ path('app_contrat_payment_success', {id: contrat.id}) }}'; }
|
||||||
|
else { formError.textContent = data.error || 'Erreur.'; formError.classList.remove('hidden'); submitBtn.disabled = false; btnText.classList.remove('hidden'); btnLoading.classList.add('hidden'); }
|
||||||
|
}).catch(function() { formError.textContent = 'Erreur de connexion.'; formError.classList.remove('hidden'); submitBtn.disabled = false; btnText.classList.remove('hidden'); btnLoading.classList.add('hidden'); });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
})();
|
||||||
|
</script>
|
||||||
|
{% endblock %}
|
||||||
54
templates/emails/contrat_setup_payment.html.twig
Normal file
54
templates/emails/contrat_setup_payment.html.twig
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
{% extends 'email/base.html.twig' %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<table width="600" cellpadding="0" cellspacing="0" style="margin: 0 auto;">
|
||||||
|
<tr>
|
||||||
|
<td style="padding: 32px;">
|
||||||
|
<h1 style="font-family: Arial, Helvetica, sans-serif; font-size: 22px; font-weight: 700; color: #111827; margin: 0 0 16px;">Chez {{ contrat.raisonSociale }},</h1>
|
||||||
|
|
||||||
|
<p style="font-family: Arial, Helvetica, sans-serif; font-size: 14px; color: #374151; line-height: 22px; margin: 0 0 16px;">
|
||||||
|
Votre contrat <strong>{{ contrat.reference }}</strong> a ete signe avec succes. Pour activer vos services, veuillez configurer votre mode de paiement et effectuer le premier reglement.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<table width="100%" cellpadding="0" cellspacing="0" style="margin: 20px 0; border: 1px solid #e5e7eb;">
|
||||||
|
<tr>
|
||||||
|
<td style="padding: 10px 16px; font-size: 11px; font-weight: 700; text-transform: uppercase; letter-spacing: 1px; color: #9ca3af; background: #f9fafb; width: 35%;">Reference</td>
|
||||||
|
<td style="padding: 10px 16px; font-family: monospace; font-size: 13px; font-weight: 700;">{{ contrat.reference }}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td style="padding: 10px 16px; font-size: 11px; font-weight: 700; text-transform: uppercase; letter-spacing: 1px; color: #9ca3af; background: #f9fafb;">Total HT / mois</td>
|
||||||
|
<td style="padding: 10px 16px; font-family: monospace; font-size: 16px; font-weight: 700; color: #fabf04;">{{ contrat.totalHt|number_format(2, ',', ' ') }} €</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
<p style="font-family: Arial, Helvetica, sans-serif; font-size: 14px; color: #374151; line-height: 22px; margin: 0 0 16px;">
|
||||||
|
Vous avez le choix entre :
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<ul style="font-family: Arial, Helvetica, sans-serif; font-size: 13px; color: #374151; line-height: 22px; margin: 0 0 16px; padding-left: 20px;">
|
||||||
|
<li style="margin-bottom: 8px;"><strong>Payer par carte bancaire</strong> : reglez votre premier mois immediatement.</li>
|
||||||
|
<li style="margin-bottom: 8px;"><strong>Configurer le prelevement SEPA</strong> : renseignez votre IBAN une seule fois, les prelevements seront automatiques chaque mois.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
{% if paymentUrl is defined and paymentUrl %}
|
||||||
|
<table cellpadding="0" cellspacing="0" style="margin: 24px auto;">
|
||||||
|
<tr>
|
||||||
|
<td style="background-color: #fabf04; padding: 14px 32px;">
|
||||||
|
<a href="{{ paymentUrl }}" style="font-family: Arial, Helvetica, sans-serif; font-size: 13px; font-weight: 700; text-transform: uppercase; letter-spacing: 1px; color: #111827; text-decoration: none;">Configurer mon paiement</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<div style="background: #fef2f2; border-left: 4px solid #dc2626; padding: 12px 16px; margin: 20px 0;">
|
||||||
|
<p style="font-family: Arial, Helvetica, sans-serif; font-size: 11px; font-weight: 700; color: #991b1b; margin: 0 0 4px;">Important</p>
|
||||||
|
<p style="font-family: Arial, Helvetica, sans-serif; font-size: 11px; color: #374151; margin: 0;">Le premier paiement doit etre effectue par carte bancaire ou prelevement SEPA. Les virements ne sont pas acceptes pour le premier paiement.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p style="font-family: Arial, Helvetica, sans-serif; font-size: 12px; color: #9ca3af; margin: 16px 0 0;">
|
||||||
|
Pour toute question : <a href="mailto:client@e-cosplay.fr" style="color: #fabf04;">client@e-cosplay.fr</a>
|
||||||
|
</p>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
{% endblock %}
|
||||||
Reference in New Issue
Block a user