前段時間因爲項目上的需求,要在.Net平臺下實現把HTML內容生成圖片或PDF文件的功能,特地在網上研究了幾種方案,這裏記錄一下以備往後再次使用。當時想着找一種開發部署都比較清爽而且運行穩定的方案,但實際上二者同時知足基本不可能,只能作一個本身以爲合適的取捨,下面從兩個維度(清爽指數和功能指數)逐一對比。html
這種方案在開發時不依賴任務外部程序集和nuget包,部署時也不須要安裝額外的工具和服務,能夠說是很是清爽了。它藉助了WinForm下的WebBrowser控件實現HTML內容渲染,並把渲染結果繪製在Bitmap中,進而保存成圖片或PDF文件。這種方案簡單粗暴,是C#中最基礎的實現方式,也是網上搜索結果最多的一種,下面看它的核心代碼(從網上拼湊得來):git
1 class WebBrowserPage2Image 2 { 3 Bitmap m_Bitmap; 4 5 string m_Url; 6 7 public void Convert(string pageUrl, string fileName) 8 { 9 m_Url = pageUrl; 10 Thread m_thread = new Thread(new ThreadStart(HtmlDrawToBitmap)); 11 m_thread.SetApartmentState(ApartmentState.STA); 12 m_thread.IsBackground = true; 13 m_thread.Start(); 14 m_thread.Join(); 15 MemoryStream stream = new MemoryStream(); 16 m_Bitmap.Save(stream, System.Drawing.Imaging.ImageFormat.Png); 17 byte[] buff = stream.ToArray(); 18 FileStream fs = new FileStream(fileName, FileMode.Create); 19 stream.WriteTo(fs); 20 stream.Dispose(); 21 stream.Close(); 22 fs.Close(); 23 } 24 25 private void HtmlDrawToBitmap() 26 { 27 WebBrowser browser = new WebBrowser(); 28 browser.ScrollBarsEnabled = false; 29 browser.Navigate(m_Url); 30 browser.DocumentCompleted += new WebBrowserDocumentCompletedEventHandler(delegate (object sender, WebBrowserDocumentCompletedEventArgs bdce) 31 { 32 if (browser.ReadyState == WebBrowserReadyState.Complete) 33 { 34 //myWebBrowser.Document.Body.Style = "zoom:180%"; 35 Rectangle r = browser.Document.Body.ScrollRectangle; 36 browser.Height = r.Height; 37 browser.Width = r.Width; 38 m_Bitmap = new Bitmap(browser.Width, browser.Height); 39 browser.BringToFront(); 40 browser.DrawToBitmap(m_Bitmap, new Rectangle() { Width = browser.Width, Height = browser.Height }); 41 } 42 }); 43 while (browser.ReadyState != WebBrowserReadyState.Complete) 44 { 45 Application.DoEvents(); 46 } 47 browser.Dispose(); 48 } 49 }
雖然開發起來很是簡潔,可是問題也很明顯。WebBrowser是Winform下的一個組件,在非Winform項目中運行會出現不可知的異常,即便在Winform項目中,數據量比較大的時候依然會出現卡死的狀況。我作過500次循環的測試,在執行到100屢次的時候程序出現假死不動也無異常拋出。除此以外,生成的圖片失真也比較嚴重,特殊字體和部分CSS樣式沒法渲染。總的來講,基本沒法達到生成環境需求。github
清爽指數:★★★★★ 功能指數:★web
這也是網上普遍流傳的一個方案,wkhtmltox是一套開源的命令行工具,提供了圖片和PDF的轉換能力,它採用C++編寫,使用Webkit做爲渲染引擎,開源地址是https://github.com/wkhtmltopdf/wkhtmltopdf。使用方法就是在命令行工具中執行命令,例如: npm
wkhtmltopdf --grayscale https://www.baidu.com baidu.pdf
若是要在.Net項目中使用的話,核心問題就是用程序喚起命令行,同時指定參數執行便可,相似於下面的代碼:api
System.Diagnostics.ProcessStartInfo Info = new System.Diagnostics.ProcessStartInfo(); Info.FileName = @"D:\dev\wkhtmltox\bin\wkhtmltopdf.exe"; Info.WindowStyle = System.Diagnostics.ProcessWindowStyle.Hidden; Info.CreateNoWindow = true; Info.Arguments = @"-q --orientation Landscape https://www.baidu.com D:\\baidu.pdf"; System.Diagnostics.Process proc = System.Diagnostics.Process.Start(Info); proc.WaitForExit(); proc.Close();
更多強大的功能例如加水印、分頁、改樣式等能夠參考這篇文章:http://www.javashuo.com/article/p-mpzgprag-bn.html瀏覽器
詳細的參數說明能夠查看文檔:https://wkhtmltopdf.org/usage/wkhtmltopdf.txtless
GitHub上有不少針對各個開發語言的封裝,使用起來比較方便,惟一不爽的是部署項目前要先安裝好這個工具。async
清爽指數:★★★★ 功能指數:★★★★ide
這個就更厲害了,說到這個就不得不先介紹下Puppeteer,由於PuppeteerSharp正是從Puppeteer衍生而來。
Puppeteer是由谷歌開源的一個Node項目,它提供了和Chrome DevTools的通訊能力,基本上咱們能在Chrome實現的操做經過它的API均可以實現,強大到讓你不敢相信。主要的應用有:
開源地址是https://github.com/GoogleChrome/puppeteer
在Node項目中使用Puppeteer很是簡單,先安裝npm包:
npm i puppeteer
安裝過程可能會有點慢,由於在安裝的時候會下載一個最近版本的Chromium(Mac下大概170M,Linux下大概282M,Windows下大概280M)。固然,若是你本地已經有一個Chromium,能夠設置npm的全局配置PUPPETEER_SKIP_CHROMIUM_DOWNLOAD
跳過下載,而後在程序中手動指定Chromium的位置。
生成圖片和PDF文件例子:
const puppeteer = require('puppeteer'); (async () => { const browser = await puppeteer.launch(); const page = await browser.newPage(); await page.goto('https://www.baidu.com'); await page.screenshot({path: 'baidu.png'}); await page.pdf({path: 'baidu.pdf', format: 'A4'}); await browser.close(); })();
Puppeteer默認使用無界面模式(headless:true),若是想看到完整的瀏覽器界面,能夠經過下面的設置開啓:
const browser = await puppeteer.launch({headless: false});
Puppeteer提供了豐富的選擇器接口,能夠輕鬆實現模擬輸入和鼠標點擊,例如:
await page.type('#index-kw', 'cnblogs');
await page.click('#index-bn');
還支持指定使用設備:
const devices = require('puppeteer/DeviceDescriptors');
await page.emulate(devices['iPhone 8']);
詳細的API文檔能夠參考:https://github.com/GoogleChrome/puppeteer/blob/master/docs/api.md
Puppeteer確實很是強大,但因爲它是一個Node包沒法直接在C#項目中使用,那怎麼辦呢?好在有國外的大神把Puppeteer移植到了.Net平臺,也就是PuppeteerSharp。
注意:PuppeteerSharp是基於NetStandard 2.0開發的,因此項目的平臺最低版本要是.NET Framework 4.6.1和.NET Core 2.0。
首先經過nuget安裝:
PM > Install-Package PuppeteerSharp
導入命名空間:
using PuppeteerSharp;
下面是我在ASP.NET Core 2.1下封裝的測試方法:
[HttpPost, Route("page2img")] public async Task<string> PageToImage(string url, int? width, int? height) { await new BrowserFetcher().DownloadAsync(BrowserFetcher.DefaultRevision); var browser = await Puppeteer.LaunchAsync(new LaunchOptions { Headless = true, //ExecutablePath="", Args = new string[] { "--no-sandbox" } }); var page = await browser.NewPageAsync(); bool fullPage = true; if (width.HasValue && height.HasValue) { await page.SetViewportAsync(new ViewPortOptions { Width = width.Value, Height = height.Value }); fullPage = false; } await page.GoToAsync(System.Web.HttpUtility.UrlDecode(url)); string fileName = $"/Files/{Guid.NewGuid().ToString()}.png"; await page.ScreenshotAsync($"{AppDomain.CurrentDomain.BaseDirectory}{fileName}", new ScreenshotOptions { FullPage = fullPage }); return $"{Request.Host.ToString()}{fileName}"; }
上面方法的第一行:
await new BrowserFetcher().DownloadAsync(BrowserFetcher.DefaultRevision);
程序會判斷本地環境有沒有可用的Chromium,若是沒有的話會自動下載一個默認版本的Chromium,這個過程可能會有點長,下載成功後會在項目根目錄多一個這樣的文件夾:
和前面說的同樣,若是本地已經下載過Chromium,能夠經過LaunchOptions的ExecutablePath字段指定一個目錄。目前PuppeteerSharp在網上的資料還不是不少,可是得益於它與Puppeteer高度完整和類似的API,Puppeteer的文檔對它基本都能適用。
整體來講,這個工具功能強大而且比較穩定(我在Windows和Linux下都測試經過),是一個不錯的選擇,可是因爲它必須依賴於Chromium來運行,打包部署並非很方便,我建議把它做爲一個獨立的web服務。
清爽指數:★★★ 功能指數:★★★★★
除了一些開源的項目和工具能提供HTML轉圖片或PDF的功能,不少商業軟件公司也提供了這樣的產品,IronPdf算是裏面比較有表明性的一個。和其餘收費軟件不一樣的是,IronPdf有一個對開發者免費試用的license:
IronPdf的主要特性包括:
咱們能夠在官網下載DLL文件直接引用到項目,也能夠經過nuget來安裝:
PM > Install-Package IronPdf
導入命名空間:
using IronPdf;
一個最簡單的例子:
// Create a PDF from any existing web page var Renderer = new IronPdf.HtmlToPdf(); Renderer.PrintOptions.EnableJavaScript = true; Renderer.PrintOptions.PaperOrientation = IronPdf.PdfPrintOptions.PdfPaperOrientation.Landscape; var PDF = Renderer.RenderUrlAsPdf("https://www.baidu.com"); PDF.SaveAs("baidu.pdf"); // This neat trick opens our PDF file so we can see the result System.Diagnostics.Process.Start("baidu.pdf");
添加水印:
pdf.WatermarkAllPages("<h2 style='color:red'>SAMPLE</h2>", PdfDocument.WaterMarkLocation.MiddleCenter, 50, -45, "https://www.baidu.com");
用圖片生成PDF文檔:
// Select one or more images. This example selects all JPEG images in a specific folder. var ImageFiles = Directory.EnumerateFiles(@"C:\project\assets").Where(f => f.EndsWith(".jpg") || f.EndsWith(".jpeg")); // Convert the images to a PDF and save it. ImageToPdfConverter.ImageToPdf(ImageFiles).SaveAs(@"C:\project\composite.pdf");
更多高級功能和配置能夠參考官網例子:https://ironpdf.com/examples/image-to-pdf/
清爽指數:★★★★ 功能指數:★★★★
以上幾種方式,都是我在本次實踐中總結出來的,可能不是很全面,歡迎你們不吝補充。
遺憾的是,最終項目沒有用上面的任何一種方式,而是抓取到HTML內容後用正則解析,而後用Bitmap一點一點從新畫圖生成圖片文件保存。由於我要截取的頁面內容不多,就是一個簡單的電子處方箋,需求上也沒有要求必須徹底和原網頁100%一致,繪圖也算是一個不錯的方案,可是缺點是一旦HTML結構或樣式發生變化,那這套東西就失效了,好在這個不會輕易變動,也算是一個折中方案。