Echeancier - Webhooks DocuSeal:
- Webhook form.completed: telecharge PDF signe + audit, state SIGNED, prepare SEPA, notifie client + admin
- Webhook form.declined: state CANCELLED, notifie client + admin
- Reference EC_ECH_XXXXX affichee dans PDF, emails, pages client, admin
- Attestation fin de paiement auto via DocuSeal au completion
Echeancier - SEPA Direct Debit (remplace Subscriptions):
- Page /echeancier/setup-payment/{id}: formulaire IBAN Stripe Elements + mandat SEPA
- Confirmation SetupIntent -> stocke PaymentMethod -> state ACTIVE
- Commande cron app:echeancier:process-payments: preleve les echeances dues via PaymentIntent off_session
- Webhooks payment_intent.succeeded/failed: met a jour EcheancierLine, notifie client
- Regularisation CB via Stripe Checkout en cas d'echec prelevement
- Bouton "Forcer prelevement" par echeance dans admin
- Infos SEPA stockees (last4, bank_code, country) + affichees admin
- Page setup_payment_done quand SEPA deja configure
- Annulation auto apres 2 rejets + sync paiements vers Advert lie
Echeancier - Lien Advert:
- Champ advert (ManyToOne nullable) sur Echeancier
- Select "Avis lie" dans formulaire creation
- AdvertPayment cree a chaque echeance payee
- Advert passe en accepted quand echeancier completed
Comptabilite:
- Export echeanciers CSV/JSON/PDF/PDF signe dans /admin/comptabilite
- Colonnes: reference, client, creance, majoration, total, paye, restant, Stripe PI, avis lie
Stats:
- Case "Total impaye global" = factures impayees + echeances non payees
- Tableau echeanciers en cours avec restant du
Confiance client:
- Statut Confiant/Attention/Danger calcule dynamiquement
- Badge en haut a droite de la fiche client
- Integre warningLevel (1st=Attention, 2nd=Attention, last=Danger)
- Creation echeancier bloquee si Danger (template + controller)
Avertissements client (tab Controle, ROLE_ROOT):
- 3 niveaux: 1st, 2nd (procedure suspension preparee), last (48h)
- Motifs cochables: impayes, irrespect, hors horaires, services gratuits
- PDF signe DocuSeal pour chaque avertissement (ClientWarningPdf)
- PDF levee avertissement signe (ClientWarningResetPdf)
- Webhooks DocuSeal client_warning + client_warning_reset
- Barre progression 4 etapes dans admin
- Mentions legales: huis clos, contestation direction@e-cosplay.fr
Cloture compte:
- Bouton "Envoyer notification de cloture" apres dernier avertissement
- PDF signe DocuSeal (ClientClosurePdf): suppression 24h, recouvrement, commissaire justice, forces ordre
- Bouton "Suspendre le compte" (state suspended)
- Webhook DocuSeal client_closure: envoie PDF signe a client + admin + direction
Factures:
- Auto-generation PDF si absent lors de l'envoi
- Bouton "Envoyer" visible meme sans PDF pour factures payees
E-Flex (financement services):
- Entites EFlex + EFlexLine (reference E_FLEX_XXXXX)
- Methodes: SEPA, CB (Stripe Checkout), virement manuel
- PDF contrat avec 2 signatures DocuSeal (Company + Client)
- Controller admin CRUD + force payment + paiement manuel
- Pages client: verify, process, sign, signed, setup SEPA, paiement CB
- Webhook DocuSeal eflex: telecharge PDFs, prepare Stripe, notifie
- Webhooks Stripe payment_intent: gestion paiements E-Flex
- Cron traite aussi les E-Flex SEPA dans process-payments
- Tab E-Flex dans fiche client avec liste + modal creation
- Emails: signature, signed, verify_code, echeance_payee, echeance_echec
Attestations custom (ROLE_ROOT):
- Entite AttestationCustom avec items JSON + HMAC SHA-256
- Repeater dynamique pour ajouter elements a attester
- PDF avec phrase officielle "Je soussigne(e)..." + QR code verification
- Signature manuelle dans DocuSeal (redirection)
- Webhook attestation_custom: telecharge PDF signe + audit
- Page publique /attestation/verify/{id}/{hmac} avec validation HMAC
- Lien dans sidebar Super Admin
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
144 lines
8.2 KiB
Twig
144 lines
8.2 KiB
Twig
{% extends 'base.html.twig' %}
|
|
|
|
{% block title %}E-Flex {{ eflex.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">Contrat E-Flex</h1>
|
|
<p class="text-xs text-white/60">{{ eflex.reference }} - Association E-Cosplay</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="p-8">
|
|
<p class="text-sm text-gray-600 mb-6">
|
|
{% if customer.raisonSociale %}Chez {{ customer.raisonSociale }}{% else %}Bonjour {{ customer.firstName }}{% endif %},
|
|
voici le detail de votre contrat de financement E-Flex.
|
|
</p>
|
|
|
|
{# Description #}
|
|
<div class="glass p-4 mb-6">
|
|
<h2 class="text-[9px] font-bold uppercase tracking-wider text-gray-400 mb-1">Service finance</h2>
|
|
<p class="text-sm font-bold">{{ eflex.description }}</p>
|
|
</div>
|
|
|
|
{# Resume #}
|
|
<div class="grid grid-cols-2 md:grid-cols-3 gap-3 mb-6">
|
|
<div class="glass p-3 text-center">
|
|
<p class="text-[9px] font-bold uppercase tracking-wider text-gray-400">Montant total</p>
|
|
<p class="text-lg font-bold mt-1">{{ eflex.totalAmount|number_format(2, ',', ' ') }} €</p>
|
|
</div>
|
|
<div class="glass p-3 text-center">
|
|
<p class="text-[9px] font-bold uppercase tracking-wider text-gray-400">Mensualite</p>
|
|
<p class="text-lg font-bold mt-1" style="color: #fabf04;">{{ eflex.monthlyAmount|number_format(2, ',', ' ') }} €</p>
|
|
</div>
|
|
<div class="glass p-3 text-center">
|
|
<p class="text-[9px] font-bold uppercase tracking-wider text-gray-400">Echeances</p>
|
|
<p class="text-lg font-bold mt-1">{{ eflex.nbLines }} mois</p>
|
|
</div>
|
|
</div>
|
|
|
|
{# Tableau echeances #}
|
|
<div class="glass overflow-hidden mb-6">
|
|
<table class="w-full text-sm">
|
|
<thead>
|
|
<tr class="glass-dark text-white">
|
|
<th class="px-4 py-2 text-left font-bold uppercase text-[10px] tracking-widest">N</th>
|
|
<th class="px-4 py-2 text-left font-bold uppercase text-[10px] tracking-widest">Date prevue</th>
|
|
<th class="px-4 py-2 text-right font-bold uppercase text-[10px] tracking-widest">Montant</th>
|
|
<th class="px-4 py-2 text-center font-bold uppercase text-[10px] tracking-widest">Statut</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{% for line in eflex.lines %}
|
|
<tr class="border-b border-white/20 {{ loop.index is odd ? 'bg-white/30' : '' }}">
|
|
<td class="px-4 py-2 font-bold">{{ line.position }}</td>
|
|
<td class="px-4 py-2 text-xs">{{ line.scheduledAt|date('d/m/Y') }}</td>
|
|
<td class="px-4 py-2 text-right font-bold">{{ line.amount|number_format(2, ',', ' ') }} €</td>
|
|
<td class="px-4 py-2 text-center">
|
|
{% if line.isPaid %}
|
|
<span class="px-2 py-0.5 bg-green-500/20 text-green-700 font-bold uppercase text-[10px]">Paye</span>
|
|
{% elseif line.isFailed %}
|
|
<span class="px-2 py-0.5 bg-red-500/20 text-red-700 font-bold uppercase text-[10px]">Echoue</span>
|
|
{% else %}
|
|
<span class="px-2 py-0.5 bg-yellow-100 text-yellow-800 font-bold uppercase text-[10px]">A payer</span>
|
|
{% endif %}
|
|
</td>
|
|
</tr>
|
|
{% endfor %}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
|
|
{# Conditions #}
|
|
<div class="text-xs text-gray-500 mb-6 space-y-1">
|
|
<p class="font-bold uppercase text-[9px] tracking-wider text-gray-400 mb-2">Conditions</p>
|
|
<p>E-Flex est une solution de financement sans frais supplementaires proposee par l'Association E-Cosplay.</p>
|
|
<p>Les prelevements seront effectues automatiquement a chaque date prevue.</p>
|
|
</div>
|
|
|
|
{# Boutons signer / refuser (si pas encore signe) #}
|
|
{% if eflex.submissionId and eflex.state == 'draft' %}
|
|
<div class="flex justify-center gap-4 mb-6">
|
|
<a href="{{ path('app_eflex_sign', {id: eflex.id}) }}"
|
|
class="px-6 py-3 bg-green-600 text-white font-bold uppercase text-xs tracking-wider hover:bg-green-700 transition-all">
|
|
Signer le contrat
|
|
</a>
|
|
</div>
|
|
{% endif %}
|
|
|
|
{# Boutons paiement (si actif ou pending_setup) #}
|
|
{% if eflex.state in ['active', 'pending_setup'] %}
|
|
<div class="flex justify-center gap-4 mb-6">
|
|
{% if eflex.paymentMethod == 'sepa' and not eflex.stripePaymentMethodId %}
|
|
<a href="{{ path('app_eflex_setup_payment', {id: eflex.id}) }}"
|
|
class="px-6 py-3 bg-green-600 text-white font-bold uppercase text-xs tracking-wider hover:bg-green-700 transition-all">
|
|
Configurer le prelevement SEPA
|
|
</a>
|
|
{% endif %}
|
|
{% for line in eflex.lines %}
|
|
{% if line.isPending or line.isFailed %}
|
|
<a href="{{ path('app_eflex_pay', {id: eflex.id, lineId: line.id}) }}"
|
|
class="px-4 py-2 bg-gray-900 text-white font-bold uppercase text-[10px] tracking-wider hover:bg-[#fabf04] hover:text-gray-900 transition-all">
|
|
Payer echeance {{ line.position }} ({{ line.amount|number_format(2, ',', ' ') }} €)
|
|
</a>
|
|
{% endif %}
|
|
{% endfor %}
|
|
</div>
|
|
{% endif %}
|
|
|
|
{# SEPA configure #}
|
|
{% if eflex.stripePaymentMethodId %}
|
|
<div class="glass p-4 mb-6 flex items-center gap-3">
|
|
<svg xmlns="http://www.w3.org/2000/svg" class="h-6 w-6 text-green-500 shrink-0" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M5 13l4 4L19 7" />
|
|
</svg>
|
|
<p class="text-sm font-bold text-green-700">Prelevement SEPA configure - IBAN **** {{ eflex.stripeSepaLast4 ?: '****' }}</p>
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if eflex.nbPaid > 0 %}
|
|
<div class="glass p-4 mb-6">
|
|
<div class="flex justify-between items-center mb-2">
|
|
<span class="text-[9px] font-bold uppercase tracking-wider text-gray-400">Progression</span>
|
|
<span class="text-sm font-bold">{{ eflex.nbPaid }}/{{ eflex.nbLines }}</span>
|
|
</div>
|
|
<div class="w-full bg-gray-200 h-2">
|
|
<div class="bg-green-500 h-2" style="width: {{ eflex.progress }}%"></div>
|
|
</div>
|
|
</div>
|
|
{% endif %}
|
|
|
|
<p class="text-center text-xs text-gray-400 mt-6">
|
|
Pour toute question : <a href="mailto:contact@e-cosplay.fr" class="font-bold" style="color: #fabf04;">contact@e-cosplay.fr</a>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
{% endblock %}
|