狠狠色丁香婷婷综合尤物/久久精品综合一区二区三区/中国有色金属学报/国产日韩欧美在线观看 - 国产一区二区三区四区五区tv

LOGO OA教程 ERP教程 模切知識交流 PMS教程 CRM教程 開發(fā)文檔 其他文檔  
 
網(wǎng)站管理員

【C#】.NET 使用QuestPDF高效地生成PDF文檔

admin
2024年5月27日 16:26 本文熱度 1072

前言

在.NET平臺中操作生成PDF的類庫有很多如常見的有iTextSharp、PDFsharp、Aspose.PDF等,我們分享一個(gè)用于生成PDF文檔的現(xiàn)代開源.NET庫:QuestPDF,本文將介紹QuestPDF并使用它快速實(shí)現(xiàn)發(fā)票PDF文檔生成功能。

QuestPDF介紹

QuestPDF 是一個(gè)用于生成 PDF 文檔的現(xiàn)代開源 .NET 庫。QuestPDF 由簡潔易用的 C# Fluent API 提供全面的布局引擎。輕松生成 PDF 報(bào)告、發(fā)票、導(dǎo)出等。

QuestPDF它提供了一個(gè)布局引擎,在設(shè)計(jì)時(shí)考慮了完整的分頁支持。與其他庫不同,它不依賴于 HTML 到 PDF 的轉(zhuǎn)換,這在許多情況下是不可靠的。

相反,它實(shí)現(xiàn)了自己的布局引擎,該引擎經(jīng)過優(yōu)化,可以滿足所有與分頁相關(guān)的要求。

QuestPDF License

分為社區(qū)版、專業(yè)版、和企業(yè)版。

項(xiàng)目源代碼

創(chuàng)建一個(gè)控制臺應(yīng)用

創(chuàng)建一個(gè)名為QuestPDFTest的控制臺應(yīng)用。

安裝QuestPDF Nuget包

搜索:QuestPDF包進(jìn)行安裝。

快速實(shí)現(xiàn)發(fā)票PDF文檔生成

創(chuàng)建InvoiceModel

namespace QuestPDFTest  
{  
    public class InvoiceModel  
    {  
  
        /// <summary>  
        /// 發(fā)票號碼  
        /// </summary>  
        public int InvoiceNumber { getset; }  
  
        /// <summary>  
        /// 發(fā)票開具日期  
        /// </summary>  
        public DateTime IssueDate { getset; }  
  
        /// <summary>  
        /// 發(fā)票到期日期  
        /// </summary>  
        public DateTime DueDate { getset; }  
  
        /// <summary>  
        /// 賣方公司名稱  
        /// </summary>  
        public string SellerCompanyName { getset; }  
  
        /// <summary>  
        /// 買方公司名稱  
        /// </summary>  
        public string CustomerCompanyName { getset; }  
  
        /// <summary>  
        /// 訂單消費(fèi)列表  
        /// </summary>  
        public List<OrderItem> OrderItems { getset; }  
  
        /// <summary>  
        /// 備注  
        /// </summary>  
        public string Comments { getset; }  
    }  
  
    public class OrderItem  
    {  
        /// <summary>  
        /// 消費(fèi)類型  
        /// </summary>  
        public string Name { getset; }  
  
        /// <summary>  
        /// 消費(fèi)金額  
        /// </summary>  
        public decimal Price { getset; }  
  
        /// <summary>  
        /// 消費(fèi)數(shù)量  
        /// </summary>  
        public int Quantity { getset; }  
    }  
}  

CreateInvoiceDetails

namespace QuestPDFTest  
{  
    public class CreateInvoiceDetails  
    {  
        private static readonly Random _random = new Random();  
  
        public enum InvoiceType  
        {  
            餐飲費(fèi),  
            交通費(fèi),  
            住宿費(fèi),  
            日用品,  
            娛樂費(fèi),  
            醫(yī)療費(fèi),  
            通訊費(fèi),  
            教育費(fèi),  
            裝修費(fèi),  
            旅游費(fèi)  
        }  
  
        /// <summary>  
        /// 獲取發(fā)票詳情數(shù)據(jù)  
        /// </summary>  
        /// <returns></returns>  
        public static InvoiceModel GetInvoiceDetails()  
        {  
            return new InvoiceModel  
            {  
                InvoiceNumber = _random.Next(1_000, 10_000),  
                IssueDate = DateTime.Now,  
                DueDate = DateTime.Now + TimeSpan.FromDays(14),  
                SellerCompanyName = "追逐時(shí)光者",  
                CustomerCompanyName = "DotNetGuide技術(shù)社區(qū)",  
                OrderItems = Enumerable  
                .Range(120)  
                .Select(_ => GenerateRandomOrderItemInfo())  
                .ToList(),  
                Comments = "DotNetGuide技術(shù)社區(qū)是一個(gè)面向.NET開發(fā)者的開源技術(shù)社區(qū),旨在為開發(fā)者們提供全面的C#/.NET/.NET Core相關(guān)學(xué)習(xí)資料、技術(shù)分享和咨詢、項(xiàng)目推薦、招聘資訊和解決問題的平臺。在這個(gè)社區(qū)中,開發(fā)者們可以分享自己的技術(shù)文章、項(xiàng)目經(jīng)驗(yàn)、遇到的疑難技術(shù)問題以及解決方案,并且還有機(jī)會(huì)結(jié)識志同道合的開發(fā)者。我們致力于構(gòu)建一個(gè)積極向上、和諧友善的.NET技術(shù)交流平臺,為廣大.NET開發(fā)者帶來更多的價(jià)值和成長機(jī)會(huì)。"  
            };  
        }  
  
        /// <summary>  
        /// 訂單信息生成  
        /// </summary>  
        /// <returns></returns>  
        private static OrderItem GenerateRandomOrderItemInfo()  
        {  
            var types = (InvoiceType[])Enum.GetValues(typeof(InvoiceType));  
            return new OrderItem  
            {  
                Name = types[_random.Next(types.Length)].ToString(),  
                Price = (decimal)Math.Round(_random.NextDouble() * 1002),  
                Quantity = _random.Next(110)  
            };  
        }  
    }  
}  

CreateInvoiceDocument

using QuestPDF.Fluent;  
using QuestPDF.Helpers;  
using QuestPDF.Infrastructure;  
  
namespace QuestPDFTest  
{  
    public class CreateInvoiceDocument : IDocument  
    {  
        /// <summary>  
        /// 獲取Logo的的Image對象  
        /// </summary>  
        public static Image LogoImage { get; } = Image.FromFile("dotnetguide.png");  
  
        public InvoiceModel Model { get; }  
  
        public CreateInvoiceDocument(InvoiceModel model)  
        {  
            Model = model;  
        }  
  
        public DocumentMetadata GetMetadata() => DocumentMetadata.Default;  
  
        public void Compose(IDocumentContainer container)  
        {  
            container  
                .Page(page =>  
                {  
                    //設(shè)置頁面的邊距  
                    page.Margin(50);  
  
                    //字體默認(rèn)大小18號字體  
                    page.DefaultTextStyle(x => x.FontSize(18));  
  
                    //頁眉部分  
                    page.Header().Element(BuildHeaderInfo);  
  
                    //內(nèi)容部分  
                    page.Content().Element(BuildContentInfo);  
  
                    //頁腳部分  
                    page.Footer().AlignCenter().Text(text =>  
                    {  
                        text.CurrentPageNumber();  
                        text.Span(" / ");  
                        text.TotalPages();  
                    });  
                });  
        }  
  
        #region 構(gòu)建頁眉部分  
        void BuildHeaderInfo(IContainer container)  
        {  
            container.Row(row =>  
            {  
                row.RelativeItem().Column(column =>  
                {  
                    column.Item().Text($"發(fā)票編號 #{Model.InvoiceNumber}").FontFamily("fangsong").FontSize(20).SemiBold().FontColor(Colors.Blue.Medium);  
  
                    column.Item().Text(text =>  
                    {  
                        text.Span("發(fā)行日期: ").FontFamily("fangsong").FontSize(13).SemiBold();  
                        text.Span($"{Model.IssueDate:d}");  
                    });  
  
                    column.Item().Text(text =>  
                    {  
                        text.Span("終止日期: ").FontFamily("fangsong").FontSize(13).SemiBold();  
                        text.Span($"{Model.DueDate:d}");  
                    });  
                });  
  
                //在當(dāng)前行的常量項(xiàng)中插入一個(gè)圖像  
                row.ConstantItem(130).Image(LogoImage);  
            });  
        }  
  
        #endregion  
  
        #region 構(gòu)建內(nèi)容部分  
  
        void BuildContentInfo(IContainer container)  
        {  
            container.PaddingVertical(40).Column(column =>  
            {  
                column.Spacing(20);  
  
                column.Item().Row(row =>  
                {  
                    row.RelativeItem().Component(new AddressComponent("賣方公司名稱", Model.SellerCompanyName));  
                    row.ConstantItem(50);  
                    row.RelativeItem().Component(new AddressComponent("客戶公司名稱", Model.CustomerCompanyName));  
                });  
  
                column.Item().Element(CreateTable);  
  
                var totalPrice = Model.OrderItems.Sum(x => x.Price * x.Quantity);  
                column.Item().PaddingRight(5).AlignRight().Text($"總計(jì): {totalPrice}").FontFamily("fangsong").SemiBold();  
  
                if (!string.IsNullOrWhiteSpace(Model.Comments))  
                    column.Item().PaddingTop(25).Element(BuildComments);  
            });  
        }  
  
        /// <summary>  
        /// 創(chuàng)建表格  
        /// </summary>  
        /// <param name="container">container</param>  
        void CreateTable(IContainer container)  
        {  
            var headerStyle = TextStyle.Default.SemiBold();  
  
            container.Table(table =>  
            {  
                table.ColumnsDefinition(columns =>  
                {  
                    columns.ConstantColumn(25);  
                    columns.RelativeColumn(3);  
                    columns.RelativeColumn();  
                    columns.RelativeColumn();  
                    columns.RelativeColumn();  
                });  
  
                table.Header(header =>  
                {  
                    header.Cell().Text("#").FontFamily("fangsong");  
                    header.Cell().Text("消費(fèi)類型").Style(headerStyle).FontFamily("fangsong");  
                    header.Cell().AlignRight().Text("花費(fèi)金額").Style(headerStyle).FontFamily("fangsong");  
                    header.Cell().AlignRight().Text("數(shù)量").Style(headerStyle).FontFamily("fangsong");  
                    header.Cell().AlignRight().Text("總金額").Style(headerStyle).FontFamily("fangsong");  
                    //設(shè)置了表頭單元格的屬性  
                    header.Cell().ColumnSpan(5).PaddingTop(5).BorderBottom(1).BorderColor(Colors.Black);  
                });  
  
                foreach (var item in Model.OrderItems)  
                {  
                    var index = Model.OrderItems.IndexOf(item) + 1;  
  
                    table.Cell().Element(CellStyle).Text($"{index}").FontFamily("fangsong");  
                    table.Cell().Element(CellStyle).Text(item.Name).FontFamily("fangsong");  
                    table.Cell().Element(CellStyle).AlignRight().Text($"{item.Price}").FontFamily("fangsong");  
                    table.Cell().Element(CellStyle).AlignRight().Text($"{item.Quantity}").FontFamily("fangsong");  
                    table.Cell().Element(CellStyle).AlignRight().Text($"{item.Price * item.Quantity}").FontFamily("fangsong");  
                    static IContainer CellStyle(IContainer container) => container.BorderBottom(1).BorderColor(Colors.Grey.Lighten2).PaddingVertical(5);  
                }  
            });  
        }  
  
        #endregion  
  
        #region 構(gòu)建頁腳部分  
  
        void BuildComments(IContainer container)  
        {  
            container.ShowEntire().Background(Colors.Grey.Lighten3).Padding(10).Column(column =>  
            {  
                column.Spacing(5);  
                column.Item().Text("DotNetGuide技術(shù)社區(qū)介紹").FontSize(14).FontFamily("fangsong").SemiBold();  
                column.Item().Text(Model.Comments).FontFamily("fangsong");  
            });  
        }  
  
        #endregion  
    }  
  
    public class AddressComponent : IComponent  
    {  
        private string Title { get; }  
        private string CompanyName { get; }  
  
        public AddressComponent(string title, string companyName)  
        {  
            Title = title;  
            CompanyName = companyName;  
        }  
  
        public void Compose(IContainer container)  
        {  
            container.ShowEntire().Column(column =>  
            {  
                column.Spacing(2);  
  
                column.Item().Text(Title).FontFamily("fangsong").SemiBold();  
                column.Item().PaddingBottom(5).LineHorizontal(1);  
                column.Item().Text(CompanyName).FontFamily("fangsong");  
            });  
        }  
    }  
}  

Program

using QuestPDF;  
using QuestPDF.Fluent;  
using QuestPDF.Infrastructure;  
  
namespace QuestPDFTest  
{  
    internal class Program  
    {  
        static void Main(string[] args)  
        {  
            // 1、請確保您有資格使用社區(qū)許可證,不設(shè)置的話會(huì)報(bào)異常。  
            Settings.License = LicenseType.Community;  
  
            // 2、禁用QuestPDF庫中文本字符可用性的檢查  
            Settings.CheckIfAllTextGlyphsAreAvailable = false;  
  
            // 3、PDF Document 創(chuàng)建  
            var invoiceSourceData = CreateInvoiceDetails.GetInvoiceDetails();  
            var document = new CreateInvoiceDocument(invoiceSourceData);  
  
            // 4、生成 PDF 文件并在默認(rèn)的查看器中顯示  
            document.GeneratePdfAndShow();  
        }  
    }  
}  

完整示例源代碼

https://github.com/YSGStudyHards/QuestPDFTest

示例運(yùn)行效果圖

注意問題

中文報(bào)異常

QuestPDF.Drawing.Exceptions.DocumentDrawingException:“Could not find an appropriate font fallback for glyph: U-53D1 '發(fā)'. Font families available on current environment that contain this glyph: Microsoft JhengHei, Microsoft JhengHei UI, Microsoft YaHei, Microsoft YaHei UI, SimSun, NSimSun, DengXian, FangSong, KaiTi, SimHei, FZCuHeiSongS-B-GB. Possible solutions: 1) Use one of the listed fonts as the primary font in your document. 2) Configure the fallback TextStyle using the 'TextStyle.Fallback' method with one of the listed fonts. You can disable this check by setting the 'Settings.CheckIfAllTextGlyphsAreAvailable' option to 'false'. However, this may result with text glyphs being incorrectly rendered without any warning.”  

加上這段代碼

// 2、禁用QuestPDF庫中文本字符可用性的檢查  
Settings.CheckIfAllTextGlyphsAreAvailable = false;  

原因:默認(rèn)情況下,使用 QuestPDF 生成 PDF 文檔時(shí),它會(huì)檢查所使用的字體是否支持文本中的所有字符,并在發(fā)現(xiàn)不能顯示的字符時(shí)輸出一條警告消息。這個(gè)選項(xiàng)可以確保文本中的所有字符都能正確地顯示在生成的 PDF 文件中。

中文亂碼問題

解決方案

假如Text("")中為漢字一定要在后面加上FontFamily("fangsong")[仿宋字體]或FontFamily("simhei")[黑體字體],否則中文無法正常顯示。

項(xiàng)目源碼地址

GitHub地址:https://github.com/QuestPDF/QuestPDF

文檔地址:https://www.questpdf.com/api-reference/

- EOF -


該文章在 2024/5/27 16:26:26 編輯過
關(guān)鍵字查詢
相關(guān)文章
正在查詢...
點(diǎn)晴ERP是一款針對中小制造業(yè)的專業(yè)生產(chǎn)管理軟件系統(tǒng),系統(tǒng)成熟度和易用性得到了國內(nèi)大量中小企業(yè)的青睞。
點(diǎn)晴PMS碼頭管理系統(tǒng)主要針對港口碼頭集裝箱與散貨日常運(yùn)作、調(diào)度、堆場、車隊(duì)、財(cái)務(wù)費(fèi)用、相關(guān)報(bào)表等業(yè)務(wù)管理,結(jié)合碼頭的業(yè)務(wù)特點(diǎn),圍繞調(diào)度、堆場作業(yè)而開發(fā)的。集技術(shù)的先進(jìn)性、管理的有效性于一體,是物流碼頭及其他港口類企業(yè)的高效ERP管理信息系統(tǒng)。
點(diǎn)晴WMS倉儲管理系統(tǒng)提供了貨物產(chǎn)品管理,銷售管理,采購管理,倉儲管理,倉庫管理,保質(zhì)期管理,貨位管理,庫位管理,生產(chǎn)管理,WMS管理系統(tǒng),標(biāo)簽打印,條形碼,二維碼管理,批號管理軟件。
點(diǎn)晴免費(fèi)OA是一款軟件和通用服務(wù)都免費(fèi),不限功能、不限時(shí)間、不限用戶的免費(fèi)OA協(xié)同辦公管理系統(tǒng)。
Copyright 2010-2025 ClickSun All Rights Reserved