using PyroFetes.Models; using QuestPDF.Companion; using QuestPDF.Fluent; using QuestPDF.Helpers; using QuestPDF.Infrastructure; namespace PyroFetes.Services.Pdf; public interface IQuotationPdfService { byte[] Generate(Quotation quotation, List lignes); } public class QuotationPdfService : IQuotationPdfService { public byte[] Generate(Quotation quotation, List lignes) { var logoPath = Path.Combine(AppContext.BaseDirectory, "wwwroot", "Images", "logo.jpg"); var signaturePath = Path.Combine(AppContext.BaseDirectory, "wwwroot", "Images", "signature.png"); int total = 0; var document = Document.Create(container => { container.Page(page => { page.Size(PageSizes.A4); page.Margin(30); page.DefaultTextStyle(x => x.FontSize(11)); page.Header().Row(row => { // Client à gauche row.RelativeItem().Column(col => { col.Item().Text(""); col.Item().Text(""); col.Item().Text(""); col.Item().Text(""); col.Item().Text("Client").SemiBold().FontSize(12); col.Item().Text($"{quotation.Customer}"); }); // Logo + société à droite row.ConstantItem(200).Column(col => { col.Item().AlignRight().Height(70).Image(logoPath, ImageScaling.FitArea); col.Item().Height(20); col.Item().AlignRight().Text("Pyro-Fêtes").SemiBold(); col.Item().Height(5); col.Item().AlignRight().Text("24, rue La Fosse Mardeau\n41700 Le Controis-en-Sologne"); col.Item().Height(5); col.Item().AlignRight().Text("Téléphone: 02 54 78 77 66"); col.Item().Height(5); col.Item().AlignRight().Text("SIRET: 82031463100012"); col.Item().Height(40); }); }); page.Content().Column(col => { // Titre + date col.Item().Row(row => { row.RelativeItem().Text($"Devis n° {quotation.Id}") .FontSize(16).SemiBold(); row.ConstantItem(200).AlignRight().Text( $"Le {DateTime.Now:dd/MM/yyyy}"); }); col.Item().Height(20); col.Item().LineHorizontal(1); // Tableau des lignes col.Item().Table(table => { table.ColumnsDefinition(columns => { columns.RelativeColumn(10); // Produit columns.RelativeColumn(2); // Qté columns.RelativeColumn(3); // PU columns.RelativeColumn(3); // Total }); // En-têtes table.Header(header => { header.Cell().Element(CellHeader).Text("Produit"); header.Cell().Element(CellHeader).AlignRight().Text("Qté"); header.Cell().Element(CellHeader).AlignRight().Text("PU"); header.Cell().Element(CellHeader).AlignRight().Text("Total"); }); foreach (var l in lignes) { table.Cell().Element(CellBody).Text(l.Product?.Name); table.Cell().Element(CellBody).AlignRight().Text(l.Quantity.ToString()); table.Cell().Element(CellBody).AlignRight().Text($"{l.Quantity:n2} €"); table.Cell().Element(CellBody).AlignRight().Text($"{l.Quantity * l.Quantity:n2} €"); total = total + l.Quantity * l.Quantity; } IContainer CellHeader(IContainer c) => c.BorderBottom(1).PaddingVertical(5).DefaultTextStyle(x => x.SemiBold()); IContainer CellBody(IContainer c) => c.PaddingVertical(2); }); col.Item().LineHorizontal(1); col.Item().Height(30); col.Item().Row(row => { // Colonne gauche : conditions de vente row.RelativeItem().Column(left => { left.Item().Text("Conditions de vente") .SemiBold().FontSize(12); left.Item().Text(quotation.ConditionsSale) .FontSize(9); }); // Colonne droite : totaux row.ConstantItem(180).Column(right => { right.Item().AlignRight().Text($"Total HT : {total:n2} €"); right.Item().AlignRight().Text("Taxe : 20 %"); right.Item().AlignRight().Text($"Total TTC : {(total * 1.2):n2} €"); }); }); }); // Signature en bas à droite page.Footer().AlignRight().Column(col => { col.Item().AlignRight().Height(100).Image(signaturePath, ImageScaling.FitArea); }); }); }); // Pour avoir la vue du PDF en temps réel // document.ShowInCompanion(); return document.GeneratePdf(); } }