Files
crm_ecosplay/src/Service/Pdf/ComptaPdf.php

410 lines
13 KiB
PHP
Raw Normal View History

feat: comptabilite + prestataires + rapport financier + stats dynamiques Comptabilite (Super Admin) : - ComptabiliteController avec 7 exports CSV/JSON compatibles SAGE (journal ventes, grand livre, FEC, balance agee, reglements, commissions Stripe 1.5%+0.25E, couts services) - Export PDF via ComptaPdf (FPDF) avec bloc legal pre-rempli, tableau pagine, champ signature DocuSeal - Signature electronique DocuSeal + callback + envoi email signe avec template dedie (compta_export_signed.html.twig) - Rapport financier public (RapportFinancierPdf) : recettes par service, depenses (Stripe, infra, prestataires), bilan excedent/deficit - Codes comptables clients EC-XXXX (plus de 411xxx) Prestataires (Super Admin) : - Entite Prestataire (raisonSociale, siret, email, phone, adresse) - Entite FacturePrestataire (numFacture, montantHt, montantTtc, year, month, isPaid, PDF via Vich) - CRUD complet avec recherche SIRET via proxy API data.gouv.fr - Commande cron app:reminder:factures-prestataire (5 du mois) - Factures prestataires integrees dans export couts services - Sidebar Super Admin : entree Prestataires + Comptabilite Stats (/admin/stats) : - Cout prestataire dynamique depuis FacturePrestataire - Fusion Infra + Prestataire en "Cout de fonctionnement" - Commission Stripe corrigee (1.5% + 0.25E par transaction) Divers : - DocuSealService::sendComptaForSignature() + getApi() - Customer::generateCodeComptable() format EC-XXXX-XXXXX - Protection double prefixe EC- a la creation client - Bouton regenerer PDF cache quand advert state=accepted - Modals sans script inline (data-modal-open/close dans app.js) Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-07 23:39:31 +02:00
<?php
namespace App\Service\Pdf;
use setasign\Fpdi\Fpdi;
use Symfony\Component\HttpKernel\KernelInterface;
if (!\defined('EURO')) {
\define('EURO', \chr(128));
}
class ComptaPdf extends Fpdi
{
private string $documentTitle;
private string $periodFrom;
private string $periodTo;
/** @var list<array<string, string>> */
private array $rows = [];
/** @var list<string> */
private array $columns = [];
/** @var array<string, int> */
private array $columnWidths = [];
public function __construct(
private readonly KernelInterface $kernel,
string $documentTitle,
string $periodFrom,
string $periodTo,
) {
parent::__construct();
$this->documentTitle = $documentTitle;
$this->periodFrom = $periodFrom;
$this->periodTo = $periodTo;
$this->SetTitle($this->enc($documentTitle));
$this->SetAuthor($this->enc('Association E-Cosplay'));
$this->SetCreator('CRM E-Cosplay');
}
/**
* @param list<array<string, string>> $rows
*/
public function setData(array $rows): void
{
$this->rows = $rows;
if (!empty($rows)) {
$this->columns = array_keys($rows[0]);
$this->columnWidths = $this->computeColumnWidths();
}
}
public function generate(): void
{
$this->AliasNbPages();
$this->AddPage('L');
$this->writeContextBlock();
$this->writeDataTable();
$this->writeSummary();
$this->writeSignatureBlock();
}
// ---------------------------------------------------------------
// Header / Footer
// ---------------------------------------------------------------
public function Header(): void
{
$logo = $this->kernel->getProjectDir().'/public/logo.jpg';
if (file_exists($logo)) {
$this->Image($logo, 10, 8, 40);
}
// Titre du document
$this->SetFont('Arial', 'B', 16);
$this->SetXY(60, 10);
$this->Cell(0, 8, $this->enc(mb_strtoupper($this->documentTitle)), 0, 1, 'L');
// Periode
$this->SetFont('Arial', '', 10);
$this->SetXY(60, 18);
$this->Cell(0, 5, $this->enc('Periode : du '.$this->periodFrom.' au '.$this->periodTo), 0, 1, 'L');
// Date de generation
$formatter = new \IntlDateFormatter(
'fr_FR',
\IntlDateFormatter::FULL,
\IntlDateFormatter::NONE,
'Europe/Paris',
\IntlDateFormatter::GREGORIAN
);
$this->SetXY(60, 23);
$this->SetFont('Arial', '', 9);
$this->SetTextColor(100, 100, 100);
$this->Cell(0, 5, $this->enc('Genere le '.$formatter->format(new \DateTime())), 0, 1, 'L');
$this->SetTextColor(0, 0, 0);
$this->Ln(5);
}
public function Footer(): void
{
$this->SetY(-20);
$this->SetDrawColor(253, 140, 4);
$this->Line(15, $this->GetY(), $this->GetPageWidth() - 15, $this->GetY());
$this->Ln(3);
$this->SetFont('Arial', '', 7);
$this->SetTextColor(0, 0, 0);
$this->Cell(0, 3, $this->enc('42 rue de Saint-Quentin - 02800 BEAUTOR - Tel: 06 79 34 88 02 - contact@e-cosplay.fr - www.e-cosplay.fr'), 0, 1, 'C');
$this->Cell(0, 3, $this->enc('Association E-Cosplay - N SIRET 943 121 517 00011 - CODE APE 9329Z - RNA W022006988'), 0, 1, 'C');
$this->SetFont('Arial', 'I', 7);
$this->SetTextColor(150, 150, 150);
$this->Cell(0, 3, $this->enc('Page ').$this->PageNo().' / {nb}', 0, 0, 'C');
}
// ---------------------------------------------------------------
// Bloc legal / contextuel
// ---------------------------------------------------------------
private function writeContextBlock(): void
{
$this->SetY(35);
$labelW = 55;
$dataW = 120;
// Ligne superieure
$this->SetDrawColor(200, 200, 200);
$this->Cell(0, 0.5, '', 'T', 1, 'L');
$this->Ln(2);
$this->SetFont('Arial', 'B', 10);
$this->Cell(0, 5, $this->enc('INFORMATIONS LEGALES ET CONTEXTUELLES'), 0, 1, 'C');
$this->Ln(1);
$this->SetFont('Arial', '', 9);
// Association / RNA
$this->Cell($labelW, 5, $this->enc('Association :'), 0, 0, 'L');
$this->SetFont('Arial', 'B', 9);
$this->Cell($dataW, 5, $this->enc('E-Cosplay Association loi 1901 - RNA N W022006988'), 0, 1, 'L');
$this->SetFont('Arial', '', 9);
// Siege social
$this->Cell($labelW, 5, $this->enc('Siege social :'), 0, 0, 'L');
$this->Cell($dataW, 5, $this->enc('42 rue de Saint-Quentin 02800 Beautor'), 0, 1, 'L');
// SIRET
$this->Cell($labelW, 5, $this->enc('SIRET :'), 0, 0, 'L');
$this->Cell($dataW, 5, $this->enc('943 121 517 00011'), 0, 1, 'L');
// Code APE
$this->Cell($labelW, 5, $this->enc('Code APE :'), 0, 0, 'L');
$this->Cell($dataW, 5, $this->enc('9329Z'), 0, 1, 'L');
// Document
$this->Cell($labelW, 5, $this->enc('Document :'), 0, 0, 'L');
$this->SetFont('Arial', 'B', 9);
$this->Cell($dataW, 5, $this->enc($this->documentTitle), 0, 1, 'L');
$this->SetFont('Arial', '', 9);
// Periode
$this->Cell($labelW, 5, $this->enc('Periode comptable :'), 0, 0, 'L');
$this->Cell($dataW, 5, $this->enc('Du '.$this->periodFrom.' au '.$this->periodTo), 0, 1, 'L');
// Ligne inferieure
$this->Ln(2);
$this->Cell(0, 0.5, '', 'T', 1, 'L');
$this->Ln(3);
}
// ---------------------------------------------------------------
// Tableau de donnees
// ---------------------------------------------------------------
private function writeDataTable(): void
{
if (empty($this->rows)) {
$this->SetFont('Arial', '', 11);
$this->Cell(0, 10, $this->enc('Aucune donnee sur cette periode.'), 0, 1, 'C');
return;
}
// En-tete du tableau
$this->SetFont('Arial', 'B', 7);
$this->SetFillColor(35, 35, 35);
$this->SetTextColor(255, 255, 255);
foreach ($this->columns as $col) {
$w = $this->columnWidths[$col] ?? 25;
$this->Cell($w, 6, $this->enc($col), 1, 0, 'C', true);
}
$this->Ln();
$this->SetTextColor(0, 0, 0);
// Donnees
$this->SetFont('Arial', '', 7);
$fill = false;
foreach ($this->rows as $row) {
// Verifier si on a besoin d'une nouvelle page
if ($this->GetY() + 5 > $this->GetPageHeight() - 30) {
$this->AddPage('L');
$this->SetY(35);
// Re-dessiner l'en-tete du tableau
$this->SetFont('Arial', 'B', 7);
$this->SetFillColor(35, 35, 35);
$this->SetTextColor(255, 255, 255);
foreach ($this->columns as $col) {
$w = $this->columnWidths[$col] ?? 25;
$this->Cell($w, 6, $this->enc($col), 1, 0, 'C', true);
}
$this->Ln();
$this->SetTextColor(0, 0, 0);
$this->SetFont('Arial', '', 7);
$fill = false;
}
$this->SetFillColor(245, 245, 240);
foreach ($this->columns as $col) {
$w = $this->columnWidths[$col] ?? 25;
$value = $row[$col] ?? '';
$align = $this->isNumericColumn($col) ? 'R' : 'L';
$this->Cell($w, 5, $this->enc($value), 'B', 0, $align, $fill);
}
$this->Ln();
$fill = !$fill;
}
$this->Ln(3);
}
// ---------------------------------------------------------------
// Resume / totaux
// ---------------------------------------------------------------
private function writeSummary(): void
{
$this->SetFont('Arial', 'B', 9);
$this->Cell(0, 6, $this->enc('Total : '.\count($this->rows).' ecriture(s)'), 0, 1, 'L');
// Si colonnes Debit/Credit, afficher les totaux
$totalDebit = 0.0;
$totalCredit = 0.0;
$hasDebitCredit = false;
foreach ($this->rows as $row) {
if (isset($row['Debit'])) {
$totalDebit += (float) $row['Debit'];
$hasDebitCredit = true;
}
if (isset($row['Credit'])) {
$totalCredit += (float) $row['Credit'];
}
// Grand livre / balance
if (isset($row['MontantHT'])) {
$totalDebit += (float) $row['MontantHT'];
}
}
if ($hasDebitCredit) {
$this->SetFont('Arial', '', 9);
$this->Cell(60, 5, $this->enc('Total Debit : '.number_format($totalDebit, 2, ',', ' ')).' '.EURO, 0, 0, 'L');
$this->Cell(60, 5, $this->enc('Total Credit : '.number_format($totalCredit, 2, ',', ' ')).' '.EURO, 0, 1, 'L');
}
$this->Ln(5);
}
// ---------------------------------------------------------------
// Bloc signature (champ DocuSeal)
// ---------------------------------------------------------------
private function writeSignatureBlock(): void
{
// S'assurer qu'on a assez de place
if ($this->GetY() + 45 > $this->GetPageHeight() - 25) {
$this->AddPage('L');
$this->SetY(35);
}
$this->SetDrawColor(200, 200, 200);
$this->Cell(0, 0.5, '', 'T', 1, 'L');
$this->Ln(3);
$formatter = new \IntlDateFormatter(
'fr_FR',
\IntlDateFormatter::LONG,
\IntlDateFormatter::NONE,
'Europe/Paris',
\IntlDateFormatter::GREGORIAN
);
$this->SetFont('Arial', '', 9);
$this->Cell(0, 5, $this->enc('Fait a Beautor, le '.$formatter->format(new \DateTime())), 0, 1, 'L');
$this->Ln(2);
$this->SetFont('Arial', 'B', 9);
$this->Cell(0, 5, $this->enc('Signature du responsable :'), 0, 1, 'L');
$this->Ln(1);
// Champ signature DocuSeal (meme format que DevisPdf)
$this->SetAutoPageBreak(false);
$this->SetFont('Arial', '', 10);
$this->SetTextColor(0, 0, 0);
$this->Cell(60, 20, '{{Sign;type=signature;role=First Party}}', 0, 1, 'L');
}
// ---------------------------------------------------------------
// Helpers
// ---------------------------------------------------------------
/**
* @return array<string, int>
*/
private function computeColumnWidths(): array
{
if (empty($this->columns)) {
return [];
}
$pageWidth = 277; // A4 landscape - margins
$widths = [];
// Colonnes connues avec taille fixe
$known = [
'JournalCode' => 15,
'JournalLib' => 28,
'EcritureNum' => 22,
'EcritureDate' => 20,
'CompteNum' => 25,
'CompteLib' => 35,
'CompAuxNum' => 22,
'CompAuxLib' => 30,
'PieceRef' => 22,
'PieceDate' => 20,
'EcritureLib' => 45,
'Debit' => 18,
'Credit' => 18,
'Lettrage' => 14,
'DateLettrage' => 20,
'ValidDate' => 20,
'MontantDevise' => 18,
'Idevise' => 12,
'EcrtureLet' => 14,
'DateLet' => 18,
'Montantdevise' => 18,
'Solde' => 18,
'CodeJournal' => 15,
'Statut' => 16,
'MethodePaiement' => 24,
'DatePaiement' => 20,
'CodeComptable' => 25,
'Client' => 40,
'NumFacture' => 25,
'DateFacture' => 20,
'MontantHT' => 20,
'MontantTVA' => 20,
'MontantTTC' => 20,
'JoursRetard' => 16,
'Tranche' => 22,
'DateReglement' => 22,
'CompteBanque' => 20,
];
$usedWidth = 0;
$unknownCols = [];
foreach ($this->columns as $col) {
if (isset($known[$col])) {
$widths[$col] = $known[$col];
$usedWidth += $known[$col];
} else {
$unknownCols[] = $col;
}
}
// Distribuer l'espace restant aux colonnes inconnues
if (!empty($unknownCols)) {
$remaining = max($pageWidth - $usedWidth, \count($unknownCols) * 15);
$each = (int) floor($remaining / \count($unknownCols));
foreach ($unknownCols as $col) {
$widths[$col] = $each;
}
}
return $widths;
}
private function isNumericColumn(string $col): bool
{
return \in_array($col, [
'Debit', 'Credit', 'Solde',
'MontantHT', 'MontantTVA', 'MontantTTC',
'MontantDevise', 'Montantdevise',
'JoursRetard',
], true);
}
private function enc(string $text): string
{
return mb_convert_encoding($text, 'Windows-1252', 'UTF-8');
}
}