Word 转成pdf及打印的开源方案支持xp

server/2025/1/14 20:20:07/

Word转成pdf、打印的方案几乎没有免费开源的方案,现在提供一个通过LibreOffice实现的方案

操作依赖LibreOffice需要安装,点此下载老版本

5.4.7.2是最后一个支持xp的 版本如需xp要请安装此版本

LibreOffice官方介绍

LibreOffice 是一款开放源代码的自由免费全能办公软件,可运行于 Microsoft Windows, GNU/Linux 以及 macOS 等操作系统上。它包含了 Writer, Calc, Impress, Draw, Math 以及 Base 等组件,可分别用于文本文档、电子表格、幻灯片演示文稿、绘图文档、数学公式编辑、数据库管理等工作。

LibreOffice 采用对企业和个人用户均免费的 MPL 2.0 授权协议。您可以自由分发该软件,无需支付授权费用(但您仍然可以付费获得经认证的专业支持)。它的源代码完全公开,任何人都可以参与软件的开发和维护。

一、通过进程的方式

1.Word打印

  public void PrintWordFile(string file, string printerName){if (string.IsNullOrEmpty(file)) throw new Exception("Invalid parameters passed to convert word function.");if (!File.Exists(file)) throw new FileNotFoundException($"The file passed to the convert word process ({file}) could not be found.");var fileInfo = new FileInfo(file);if (fileInfo.Extension.ToLower() != ".doc" && fileInfo.Extension.ToLower() != ".docx")throw new ArgumentOutOfRangeException($"The file type passed to the convert word process is an invalid type ({fileInfo.Extension}).");var libreOfficePath = Path.Combine(LibreOfficePath, "swriter.exe");if (!File.Exists(libreOfficePath)) throw new FileNotFoundException("It seems that LibreOffice is not where it should be, please ensure the path exists.");var procStartInfo = new ProcessStartInfo(libreOfficePath, $@"--headless --pt ""{printerName}"" ""{file}"""){RedirectStandardOutput = true,RedirectStandardError = true,UseShellExecute = false,CreateNoWindow = true,WorkingDirectory = Environment.CurrentDirectory};Process process = new Process() { StartInfo = procStartInfo };// Attach event handlers to capture output and error streamsprocess.OutputDataReceived += (sender, args) => Console.WriteLine("OUT: " + args.Data);process.ErrorDataReceived += (sender, args) => Console.WriteLine("ERR: " + args.Data);process.Start();// Start reading the output asynchronouslyprocess.BeginOutputReadLine();process.BeginErrorReadLine();process.WaitForExit();if (process.ExitCode != 0){throw new LibreOfficeFailedException(process.ExitCode);}}

2.Word转成PDF

  public string ConvertWordFile(string file, string outputDirectory){if (string.IsNullOrEmpty(file) || string.IsNullOrEmpty(outputDirectory)) throw new Exception("Invalid parameters passed to convert word function.");if (!File.Exists(file)) throw new FileNotFoundException($"The file passed to the convert word process ({file}) could not be found.");if (!Directory.Exists(outputDirectory)) throw new DirectoryNotFoundException($"The output folder passed to the convert word process ({outputDirectory}) does not exist.");if (outputDirectory.EndsWith(@"\")) outputDirectory = outputDirectory.TrimEnd('\\');var fileInfo = new FileInfo(file);if (fileInfo.Extension.ToLower() == ".doc" && fileInfo.Extension.ToLower() == ".docx") throw new ArgumentOutOfRangeException($"The file type passed to the convert word process is an invalid type ({fileInfo.Extension}).");var outputFile = outputDirectory + @"\" + Path.GetFileNameWithoutExtension(fileInfo.Name) + ".pdf";if (File.Exists(outputFile)) File.Delete(outputFile);var libreOfficePath = Path.Combine(LibreOfficePath, "swriter.exe");if (!File.Exists(libreOfficePath)) throw new FileNotFoundException("It seems that LibreOffice is not where it should be, please ensure the path exists.");var procStartInfo = new ProcessStartInfo(libreOfficePath, $@"--headless --convert-to pdf:writer_pdf_Export ""{file}"" --outdir ""{outputDirectory}"""){RedirectStandardOutput = true,UseShellExecute = false,CreateNoWindow = true,WorkingDirectory = Environment.CurrentDirectory};Process process = new Process() { StartInfo = procStartInfo };process.Start();process.WaitForExit();if (process.ExitCode != 0)throw new LibreOfficeFailedException(process.ExitCode);if (!File.Exists(outputFile)) throw new FileNotFoundException("The convert to word process has failed to convert the file!");return outputFile;}public class LibreOfficeFailedException : Exception{public LibreOfficeFailedException(int exitCode) : base($"LibreOffice has failed with {exitCode}") { }}

二、通过cli库调用,下附代码

CLI下载

   public class WordPrint{private XComponentContext context;private XMultiServiceFactory service;private XComponentLoader component;private XComponent doc;private static WordPrint wordPrint;private List<string> filters = new List<string>();#region ConstructorsWordPrint(){NativeMethods.InitializationLibrary();/// This will start a new instance of OpenOffice.org if it is not running, /// or it will obtain an existing instance if it is already open.context = uno.util.Bootstrap.bootstrap();/// The next step is to create a new OpenOffice.org service managerservice = (XMultiServiceFactory)context.getServiceManager();/// Create a new Desktop instance using our service managercomponent = (XComponentLoader)service.createInstance("com.sun.star.frame.Desktop");// Getting filtersXNameContainer filters = (XNameContainer)service.createInstance("com.sun.star.document.FilterFactory");foreach (string filter in filters.getElementNames())this.filters.Add(filter);}~MiniWordPrint(){if (doc != null)doc.dispose();doc = null;}#endregion#region Private methodsprivate string FilterToString(ExportFilter filter){switch (filter){case ExportFilter.Word97: return "MS Word 97";case ExportFilter.WriterPDF: return "writer_pdf_Export";case ExportFilter.CalcPDF: return "calc_pdf_Export";case ExportFilter.DrawPDF: return "draw_pdf_Export";case ExportFilter.ImpressPDF: return "impress_pdf_Export";case ExportFilter.MathPDF: return "math_pdf_Export";}return "";}#endregion#region Public methods/// <summary>/// load docx/// </summary>/// <param name="filename">file path</param>/// <param name="hidden">load document invisible Defines if the loaded component is made visible. If this property is not specified, the component is made visible by default.</param>/// <returns></returns>public bool Load(string filename, bool hidden){return Load(filename, hidden, "", "");}/// <summary>/// load docx/// </summary>/// <param name="filename">file path</param>/// <param name="hidden">load document invisible Defines if the loaded component is made visible. If this property is not specified, the component is made visible by default. </param>/// <param name="filter_index">internal filter name <see cref="Filters"/> Filters index/// Name of a filter that should be used for loading or storing the component.Names must match the names of the TypeDetection configuration, invalid names are ignored.If a name is specified on loading, it still will be verified by a filter detection, but in case of doubt it will be preferred.</param>/// <param name="filter_options">additional properties for filter/// Some filters need additional parameters; use only together with property MediaDescriptor::FilterName.Details must be documented by the filter. This is an old format for some filters. If a string is not enough, filters can use the property MediaDescriptor::FilterData.</param>/// <returns></returns>public bool Load(string filename, bool hidden, int filter_index, string filter_options){return Load(filename, hidden, filters[filter_index], filter_options);}/// <summary>/// load docx/// </summary>/// <param name="filename">file path</param>/// <param name="hidden">load document invisible Defines if the loaded component is made visible. If this property is not specified, the component is made visible by default.</param>/// <param name="filter_name">internal filter name <see cref="Filters"/>/// Name of a filter that should be used for loading or storing the component.Names must match the names of the TypeDetection configuration, invalid names are ignored.If a name is specified on loading, it still will be verified by a filter detection, but in case of doubt it will be preferred.</param>/// <param name="filter_options"> additional properties for filter/// Some filters need additional parameters; use only together with property MediaDescriptor::FilterName.Details must be documented by the filter. This is an old format for some filters. If a string is not enough, filters can use the property MediaDescriptor::FilterData.</param>/// <returns></returns>public bool Load(string filename, bool hidden, string filter_name, string filter_options){List<PropertyValue> pv = new List<PropertyValue>();pv.Add(new PropertyValue("Hidden", 0, new uno.Any(hidden), PropertyState.DIRECT_VALUE));if (filter_name != ""){pv.Add(new PropertyValue("FilterName", 0, new uno.Any(filter_name), PropertyState.DIRECT_VALUE));pv.Add(new PropertyValue("FilterOptions", 0, new uno.Any(filter_options), PropertyState.DIRECT_VALUE));}try{doc = component.loadComponentFromURL("file:///" + filename.Replace('\\', '/'), "_blank",0, pv.ToArray());return true;}catch{doc = null;return false;}}/// <summary>///  a given document xDoc to print to the standard printer without any settings/// </summary>/// <returns></returns>public bool Print(){return Print("", 1, "");}/// <summary>/// a given document xDoc to print /// </summary>/// <param name="printName">string - Specifies the name of the printer queue to be used.</param>/// <param name="copies">short - Specifies the number of copies to print.</param>/// <param name="pages">string - Specifies the pages to print in the same format as in the print dialog of the GUI (e.g. "1, 3, 4-7, 9-")</param>/// <param name="orientation">com.sun.star.view.PaperOrientation. Specifies the orientation of the paper.</param>/// <param name="paperFormat">com.sun.star.view.PaperFormat. Specifies a predefined paper size or if the paper size is a user-defined size.</param>/// <returns></returns>public bool Print(string printName, int copies, string pages, MiniOrientation orientation = MiniOrientation.PORTRAIT, MiniFormat paperFormat = MiniFormat.A4){var printerDesc = new List<PropertyValue>();if (!string.IsNullOrEmpty(printName))printerDesc.Add(new PropertyValue("Name", 0, new uno.Any(typeof(string), printName), PropertyState.DIRECT_VALUE));printerDesc.Add(new PropertyValue("PaperOrientation", 0, new uno.Any(typeof(PaperOrientation), orientation), PropertyState.DIRECT_VALUE));printerDesc.Add(new PropertyValue("PaperFormat", 0, new uno.Any(typeof(PaperFormat), paperFormat), PropertyState.DIRECT_VALUE));var printOpts = new List<PropertyValue>();printOpts.Add(new PropertyValue("CopyCount", 0, new uno.Any(copies), PropertyState.DIRECT_VALUE));if (!string.IsNullOrEmpty(pages))printOpts.Add(new PropertyValue("Pages", 0, new uno.Any(pages), PropertyState.DIRECT_VALUE));printOpts.Add(new PropertyValue("Wait", 0, new uno.Any(true), PropertyState.DIRECT_VALUE));//if (doc is XPrintable)try{((XPrintable)doc).setPrinter(printerDesc.ToArray());((XPrintable)doc).print(printOpts.ToArray());return true;}catch { return false; }}/// <summary>/// a given document xDoc to print with custom/// </summary>/// <param name="printerDesc">///----------- Properties of com.sun.star.view.PrinterDescriptor--------///*** Name string - Specifies the name of the printer queue to be used.///*** PaperOrientation com.sun.star.view.PaperOrientation.Specifies the orientation of the paper.///*** PaperFormat com.sun.star.view.PaperFormat.Specifies a predefined paper size or if the paper size is a user-defined size.///*** PaperSize com.sun.star.awt.Size.Specifies the size of the paper in 1/100 mm.///*** IsBusy boolean - Indicates if the printer is busy.///*** CanSetPaperOrientation boolean - Indicates if the printer allows changes to.PaperOrientation///*** CanSetPaperFormat boolean - Indicates if the printer allows changes to.PaperFormat///*** CanSetPaperSize boolean - Indicates if the printer allows changes to.PaperSize/// </param>/// <param name="printerDesc">///------------- Properties of com.sun.star.view.PrintOptions--------///CopyCount short - Specifies the number of copies to print.///FileName string - Specifies the name of a file to print to, if set.///Collate boolean - Advises the printer to collate the pages of the copies.If true, a whole document is printed prior to the next copy, otherwise the page copies are completed together.///Pages string - Specifies the pages to print in the same format as in the print dialog of the GUI (e.g. "1, 3, 4-7, 9-")///Wait boolean - Advises that the print job should be performed synchronously, i.e.wait until printing is complete before returning from printing.Otherwise return is immediate and following actions(e.g.closing the corresponding model) may fail until printing is complete.Default is false./// </param>/// <param name="pagePrintSettings">/// ------------- Properties of com.sun.star.text.PagePrintSettings--------/// PageRows short - Number of rows in which document pages should appear on the output page./// PageColumns short - Number of columns in which document pages should appear on the output page./// LeftMargin long - Left margin on the output page./// RightMargin long - Right margin on the output page./// TopMargin long - Top margin on the output page./// BottomMargin long - Bottom margin on the output page./// HoriMargin long - Margin between the columns on the output page./// VertMargin long - Margin between the rows on the output page./// IsLandscape boolean - Determines if the output page is in landscape format./// </param>/// <returns></returns>public bool Print(List<MiniPropertyValue> printerDesc, List<MiniPropertyValue> printOpts, List<MiniPropertyValue> pagePrintSettings){try{var printSettings = pagePrintSettings.ConvertAll(v => ToPropertyValue(v));var desc = printerDesc.ConvertAll(v => ToPropertyValue(v));var opts = printOpts.ConvertAll(v => ToPropertyValue(v));((XPagePrintable)doc).setPagePrintSettings(printSettings.ToArray());((XPrintable)doc).setPrinter(desc.ToArray());((XPrintable)doc).print(opts.ToArray());return true;}catch { return false; }}/// <summary>/// save pdf/// </summary>/// <param name="filename">file path</param>/// <param name="filter"><see cref="ExportFilter"/></param>/// <returns></returns>public bool Save(string filename, ExportFilter filter){return Save(filename, FilterToString(filter));}/// <summary>/// save pdf/// </summary>/// <param name="filename">file path</param>/// <param name="filter">/// internal filter name <see cref="Filters"/>/// Name of a filter that should be used for loading or storing the component.Names must match the names of the TypeDetection configuration, invalid names are ignored.If a name is specified on loading, it still will be verified by a filter detection, but in case of doubt it will be preferred./// </param>/// <returns></returns>public bool Save(string filename, string filter){List<PropertyValue> pv = new List<PropertyValue>();pv.Add(new PropertyValue("FilterName", 0, new uno.Any(filter), PropertyState.DIRECT_VALUE));pv.Add(new PropertyValue("Overwrite", 0, new uno.Any(true), PropertyState.DIRECT_VALUE));try{filename = filename.Replace("\\", "/");((XStorable)doc).storeToURL("file:///" + filename, pv.ToArray());return true;}catch { return false; }}/// <summary>/// export pdf/// </summary>/// <param name="filename">file path</param>/// <returns></returns>public bool ExportToPdf(string filename){filename = Path.ChangeExtension(filename, ".pdf");bool ret = Save(filename, "writer_pdf_Export");if (!ret) ret = Save(filename, "impress_pdf_Export");if (!ret) ret = Save(filename, "calc_pdf_Export");if (!ret) ret = Save(filename, "draw_pdf_Export");if (!ret) ret = Save(filename, "impress_pdf_Export");if (!ret) ret = Save(filename, "math_pdf_Export");return ret;}/// <summary>/// close XComponent /// doc stream must be not use,otherwise dispose() has no return/// </summary>public void Close(){doc.dispose();doc = null;}/// <summary>///  new docx/// </summary>/// <param name="app"><see cref="AppType"/></param>/// <param name="hidden">load document invisible Defines if the loaded component is made visible. If this property is not specified, the component is made visible by default.</param>/// <returns></returns>public bool New(AppType app, bool hidden){try{string sapp = "private:factory/";switch (app){case AppType.Writer:sapp += "swriter";break;case AppType.Calc:sapp += "scalc";break;case AppType.Impress:sapp += "simpress";break;case AppType.Draw:sapp += "sdraw";break;case AppType.Math:sapp += "smath";break;}PropertyValue pv = new PropertyValue("Hidden", 0, new uno.Any(hidden), PropertyState.DIRECT_VALUE);doc = component.loadComponentFromURL(sapp, "_blank", 0, new PropertyValue[1] { pv });return true;}catch{doc = null;return false;}}#endregion#region Properties/// <summary>/// internal filter name/// Name of a filter that should be used for loading or storing the component.Names must match the names of the TypeDetection configuration, invalid names are ignored.If a name is specified on loading, it still will be verified by a filter detection, but in case of doubt it will be preferred./// </summary>public List<string> Filters{get { return filters; }}#endregionprivate PropertyValue ToPropertyValue(MiniPropertyValue miniProperty){return new PropertyValue(Name: miniProperty.Name,Handle: miniProperty.Handle,Value: new uno.Any(type: miniProperty.Value.Type,value: miniProperty.Value.Value),State: (PropertyState)miniProperty.State);}}


http://www.ppmy.cn/server/158373.html

相关文章

C#中的常用集合

目录 一、动态数组ArrayList 二、List 三、栈&#xff08;Stack&#xff09; 四、队列&#xff08;Queue&#xff09; 五、字典&#xff08;Dictionary&#xff09;,int> 一、动态数组ArrayList ArrayList 是 C# 中提供的一种动态数组类&#xff0c;位于命名空间 Syste…

基于python Numpy的24位音频数据读取实例解析

一 概念 24位PCM编码是一种比较少见的音频编码格式&#xff0c;它采用了更高的分辨率来表达音频信号。每个采样点用3个字节&#xff08;24位&#xff09;的无符号整数表示&#xff0c;取值范围在0到2^24-1之间。这意味着它可以表达更大的动态范围和更细微的音频细节。但是&…

ubuntu 下生成 core dump

在Ubuntu下,发现程序崩溃后不生成core dump文件, 即使设置了ulimit -c unlimited后仍然无效。 1.ulimit -c unlimited 输出的的含义是核心转储文件的大小限制,单位是blocks,默认是0,表示不生成core dump文件。 2. 重设core_pattern ulimit -c unlimited后,核心转储文件…

SpringBoot项目实战(39)--Beetl网页HTML文件中静态图片及CSS、JS文件的引用和展示

使用Beetl开发网页时&#xff0c;在网页中使用的CSS、JS、图片等静态资源需要进行适当的配置才可以展示。大致的过程如下&#xff1a; &#xff08;1&#xff09;首先Spring Security框架需要允许js、css、图片资源免授权访问。 &#xff08;2&#xff09;网站开发时&#xff0…

3D目标检测数据集——kitti数据集

KITTI官网网址&#xff1a;The KITTI Vision Benchmark Suite 下载数据集&#xff1a;The KITTI Vision Benchmark Suite KITTI数据集论文&#xff1a;CMSY9 github可视化代码&#xff1a;GitHub - kuixu/kitti_object_vis: KITTI Object Visualization (Birdview, Volumetric …

基于多Agent串联与大模型理解能力的自动驾驶系统

基于多Agent串联与大模型理解能力的自动驾驶系统 摘要: 本文将详细介绍一种基于多Agent串联与大模型理解能力的自动驾驶系统。该系统利用多个具备感知、决策、规划、控制能力的Agent协同工作,并结合大语言模型的强大理解能力,实现车辆的自动驾驶功能。本文将深入探讨系统架构…

RabbitMQ-延迟交换器

在之前的延迟场景中&#xff0c;消息放入延迟队列的时间都是相同的。比如京东下单后&#xff0c;最大等待24小时进行付款。如果超过24小时还没有付款&#xff0c;那么订单将会被取消。由于下单后使用固定的时间等待。直接采用延迟队列没有任何问题。那如果是会议预订系统的提前…

如何使用 Java 的 Spring Boot 创建一个 RESTful API?

大家好&#xff0c;我是 V 哥&#xff0c;使用 Java 的 Spring Boot 创建 RESTful API 可以满足多种开发场景&#xff0c;它提供了快速开发、易于配置、可扩展、可维护的优点&#xff0c;尤其适合现代软件开发的需求&#xff0c;帮助你快速构建出高性能的后端服务。例如&#x…