added pdf generation for all orders
This commit is contained in:
@@ -0,0 +1,6 @@
|
||||
namespace PyroFetes.DTO.DeliveryNote.Request;
|
||||
|
||||
public class GetDeliveryNotePdfDto
|
||||
{
|
||||
public int Id { get; set; }
|
||||
}
|
||||
@@ -0,0 +1,6 @@
|
||||
namespace PyroFetes.DTO.PurchaseOrder.Request;
|
||||
|
||||
public class GetPurchaseOrderPdfDto
|
||||
{
|
||||
public int Id { get; set; }
|
||||
}
|
||||
@@ -0,0 +1,41 @@
|
||||
using FastEndpoints;
|
||||
using PyroFetes.DTO.DeliveryNote.Request;
|
||||
using PyroFetes.Models;
|
||||
using PyroFetes.Repositories;
|
||||
using PyroFetes.Services.Pdf;
|
||||
using PyroFetes.Specifications.DeliveryNotes;
|
||||
using PyroFetes.Specifications.Quotations;
|
||||
|
||||
namespace PyroFetes.Endpoints.DeliveryNotes;
|
||||
|
||||
public class GetDeliveryNotePdfEndpoint(
|
||||
DeliveryNotesRepository deliveryNotesRepository,
|
||||
IDeliveryNotePdfService deliveryNotePdfService)
|
||||
: Endpoint<GetDeliveryNotePdfDto>
|
||||
{
|
||||
public override void Configure()
|
||||
{
|
||||
Get("/deliveryNotes/{@Id}/pdf", x => new {x.Id});
|
||||
AllowAnonymous();
|
||||
}
|
||||
|
||||
public override async Task HandleAsync(GetDeliveryNotePdfDto req, CancellationToken ct)
|
||||
{
|
||||
DeliveryNote? deliveryNote = await deliveryNotesRepository
|
||||
.FirstOrDefaultAsync(new GetDeliveryNoteByIdWithProductsSpec(req.Id), ct);
|
||||
|
||||
if (deliveryNote == null)
|
||||
{
|
||||
await Send.NotFoundAsync(ct);
|
||||
return;
|
||||
}
|
||||
|
||||
var bytes = deliveryNotePdfService.Generate(deliveryNote, deliveryNote.ProductDeliveries!);
|
||||
|
||||
await Send.BytesAsync(
|
||||
bytes: bytes,
|
||||
contentType: "application/pdf",
|
||||
fileName: $"bon-de-livraison-{deliveryNote.Id}.pdf",
|
||||
cancellation: ct);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,39 @@
|
||||
using FastEndpoints;
|
||||
using PyroFetes.DTO.PurchaseOrder.Request;
|
||||
using PyroFetes.Models;
|
||||
using PyroFetes.Repositories;
|
||||
using PyroFetes.Services.Pdf;
|
||||
using PyroFetes.Specifications.PurchaseOrders;
|
||||
namespace PyroFetes.Endpoints.PurchaseOrders;
|
||||
|
||||
public class GetPurchaseOrderPdfEndpoint(
|
||||
PurchaseOrdersRepository purchaseOrdersRepository,
|
||||
IPurchaseOrderPdfService purchaseOrderPdfService)
|
||||
: Endpoint<GetPurchaseOrderPdfDto>
|
||||
{
|
||||
public override void Configure()
|
||||
{
|
||||
Get("/purchaseOrders/{@Id}/pdf", x => new {x.Id});
|
||||
AllowAnonymous();
|
||||
}
|
||||
|
||||
public override async Task HandleAsync(GetPurchaseOrderPdfDto req, CancellationToken ct)
|
||||
{
|
||||
PurchaseOrder? purchaseOrder = await purchaseOrdersRepository
|
||||
.FirstOrDefaultAsync(new GetPurchaseOrderByIdWithProductsSpec(req.Id), ct);
|
||||
|
||||
if (purchaseOrder == null)
|
||||
{
|
||||
await Send.NotFoundAsync(ct);
|
||||
return;
|
||||
}
|
||||
|
||||
var bytes = purchaseOrderPdfService.Generate(purchaseOrder, purchaseOrder.PurchaseProducts!);
|
||||
|
||||
await Send.BytesAsync(
|
||||
bytes: bytes,
|
||||
contentType: "application/pdf",
|
||||
fileName: $"bon-de-commande-{purchaseOrder.Id}.pdf",
|
||||
cancellation: ct);
|
||||
}
|
||||
}
|
||||
@@ -9,7 +9,6 @@ namespace PyroFetes.Endpoints.Quotations;
|
||||
|
||||
public class GetQuotationPdfEndpoint(
|
||||
QuotationsRepository quotationRepository,
|
||||
QuotationProductsRepository quotationProductRepository,
|
||||
IQuotationPdfService quotationPdfService)
|
||||
: Endpoint<GetQuotationPdfDto>
|
||||
{
|
||||
|
||||
@@ -51,6 +51,8 @@ builder.Services.AddScoped<SettingsRepository>();
|
||||
builder.Services.AddScoped<UsersRepository>();
|
||||
builder.Services.AddScoped<WarehouseProductsRepository>();
|
||||
|
||||
builder.Services.AddScoped<IDeliveryNotePdfService, DeliveryNotePdfService>();
|
||||
builder.Services.AddScoped<IPurchaseOrderPdfService, PurchaseOrderPdfService>();
|
||||
builder.Services.AddScoped<IQuotationPdfService, QuotationPdfService>();
|
||||
|
||||
MapperConfiguration mappingConfig = new(mc =>
|
||||
|
||||
133
PyroFetes/Services/Pdf/DeliveryNotePdfService.cs
Normal file
133
PyroFetes/Services/Pdf/DeliveryNotePdfService.cs
Normal file
@@ -0,0 +1,133 @@
|
||||
using PyroFetes.Models;
|
||||
using QuestPDF.Fluent;
|
||||
using QuestPDF.Helpers;
|
||||
using QuestPDF.Infrastructure;
|
||||
|
||||
namespace PyroFetes.Services.Pdf;
|
||||
|
||||
public interface IDeliveryNotePdfService
|
||||
{
|
||||
byte[] Generate(DeliveryNote deliveryNote, List<ProductDelivery> lignes);
|
||||
}
|
||||
|
||||
public class DeliveryNotePdfService : IDeliveryNotePdfService
|
||||
{
|
||||
public byte[] Generate(DeliveryNote deliveryNote, List<ProductDelivery> lignes)
|
||||
{
|
||||
var logoPath = Path.Combine(AppContext.BaseDirectory, "wwwroot", "Images", "logo.jpg");
|
||||
var signaturePath = Path.Combine(AppContext.BaseDirectory, "wwwroot", "Images", "signature.png");
|
||||
int total = 0;
|
||||
int totalQuantity = 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("Transporteur").SemiBold().FontSize(12);
|
||||
col.Item().Text($"{deliveryNote.Deliverer?.Transporter}");
|
||||
col.Item().Height(5);
|
||||
col.Item().AlignLeft().Text($"Expédiée le {deliveryNote.ExpeditionDate}");
|
||||
col.Item().Height(5);
|
||||
col.Item().AlignLeft().Text($"Estimée au {deliveryNote.EstimateDeliveryDate}");
|
||||
col.Item().Height(5);
|
||||
col.Item().AlignLeft().Text($"Reçu le {deliveryNote.RealDeliveryDate}");
|
||||
});
|
||||
|
||||
// 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($"Bon de livraison n° {deliveryNote.TrackingNumber}")
|
||||
.FontSize(16).SemiBold();
|
||||
});
|
||||
col.Item().Height(20);
|
||||
|
||||
col.Item().LineHorizontal(1);
|
||||
|
||||
// Tableau des lignes
|
||||
col.Item().Table(table =>
|
||||
{
|
||||
table.ColumnsDefinition(columns =>
|
||||
{
|
||||
columns.RelativeColumn(4); // Produit
|
||||
columns.RelativeColumn(2); // Qté
|
||||
columns.RelativeColumn(2); // PU
|
||||
columns.RelativeColumn(2); // 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} €");
|
||||
|
||||
totalQuantity += l.Quantity;
|
||||
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().AlignRight().Text($"Total: {totalQuantity:n2} produits");
|
||||
col.Item().AlignRight().Text($"Total HT: {total:n2} €");
|
||||
col.Item().AlignRight().Text("Taxe : 20 %");
|
||||
col.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);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
return document.GeneratePdf();
|
||||
}
|
||||
}
|
||||
146
PyroFetes/Services/Pdf/PurchaseOrderPdfService.cs
Normal file
146
PyroFetes/Services/Pdf/PurchaseOrderPdfService.cs
Normal file
@@ -0,0 +1,146 @@
|
||||
using PyroFetes.Models;
|
||||
using QuestPDF.Fluent;
|
||||
using QuestPDF.Helpers;
|
||||
using QuestPDF.Infrastructure;
|
||||
|
||||
namespace PyroFetes.Services.Pdf;
|
||||
|
||||
public interface IPurchaseOrderPdfService
|
||||
{
|
||||
byte[] Generate(PurchaseOrder purchaseOrder, List<PurchaseProduct> lignes);
|
||||
}
|
||||
|
||||
public class PurchaseOrderPdfService : IPurchaseOrderPdfService
|
||||
{
|
||||
public byte[] Generate(PurchaseOrder purchaseOrder, List<PurchaseProduct> lignes)
|
||||
{
|
||||
var logoPath = Path.Combine(AppContext.BaseDirectory, "wwwroot", "Images", "logo.jpg");
|
||||
var signaturePath = Path.Combine(AppContext.BaseDirectory, "wwwroot", "Images", "signature.png");
|
||||
int totalQuantity = 0;
|
||||
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("Fournisseur").SemiBold().FontSize(12);
|
||||
col.Item().Text("Mettre fournisseur ici");
|
||||
});
|
||||
|
||||
// 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($"Bon de commande n° {purchaseOrder.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(4); // Produit
|
||||
columns.RelativeColumn(1); // Qté
|
||||
columns.RelativeColumn(2); // PU
|
||||
columns.RelativeColumn(2); // 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} €");
|
||||
|
||||
totalQuantity += l.Quantity;
|
||||
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(purchaseOrder.PurchaseConditions)
|
||||
.FontSize(9);
|
||||
});
|
||||
|
||||
// Colonne droite : totaux
|
||||
row.ConstantItem(180).Column(right =>
|
||||
{
|
||||
right.Item().AlignRight().Text($"Total: {totalQuantity:n2} produits");
|
||||
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);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
return document.GeneratePdf();
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,15 @@
|
||||
using Ardalis.Specification;
|
||||
using PyroFetes.Models;
|
||||
|
||||
namespace PyroFetes.Specifications.DeliveryNotes;
|
||||
|
||||
public class GetDeliveryNoteByIdWithProductsSpec : Specification<DeliveryNote>
|
||||
{
|
||||
public GetDeliveryNoteByIdWithProductsSpec(int deliveryNoteId)
|
||||
{
|
||||
Query
|
||||
.Where(d => d.Id == deliveryNoteId)
|
||||
.Include(d => d.ProductDeliveries!)
|
||||
.ThenInclude(dp => dp.Product);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,15 @@
|
||||
using Ardalis.Specification;
|
||||
using PyroFetes.Models;
|
||||
|
||||
namespace PyroFetes.Specifications.PurchaseOrders;
|
||||
|
||||
public class GetPurchaseOrderByIdWithProductsSpec : Specification<PurchaseOrder>
|
||||
{
|
||||
public GetPurchaseOrderByIdWithProductsSpec(int purchaseOrderId)
|
||||
{
|
||||
Query
|
||||
.Where(p => p.Id == purchaseOrderId)
|
||||
.Include(p => p.PurchaseProducts!)
|
||||
.ThenInclude(pp => pp.Product);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user