【WPF.NET开发】以编程方式打印XPS文件

2024-01-20 14:52

本文主要是介绍【WPF.NET开发】以编程方式打印XPS文件,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文内容

可以使用 AddJob 方法的一个重载来打印 XML 纸张规范 (XPS) 文件,而根本无需打开 PrintDialog 或任何用户界面 (UI)(从原理上讲)。

还还可以使用多种 XpsDocumentWriter.Write 和 XpsDocumentWriter.WriteAsync 方法打印 XPS 文件。 

打印 XPS 的另一种方法是使用 PrintDialog.PrintDocument 或 PrintDialog.PrintVisual 方法。 

示例

使用三参数 AddJob(String, String, Boolean) 方法的主要步骤如下。 以下示例提供了详细信息。

  1. 确定打印机是否是 XPSDrv 打印机。 

  2. 如果打印机不是 XPSDrv 打印机,将线程的单元设置为单线程。

  3. 实例化打印服务器并打印队列对象。

  4. 调用该方法,指定作业的名称、要打印的文件和一个 Boolean 标志,该标志指示该打印机是否是 XPSDrv 打印机。

以下示例演示如何以批处理方式打印目录中的所有 XPS 文件。 尽管应用程序会提示用户指定目录,但三参数 AddJob(String, String, Boolean) 方法不需要用户界面 (UI)。 它可用于具有 XPS 文件名的任何代码路径和可以传递到该方法的路径。

只要 Boolean 参数为 false(使用非 XPSDrv 打印机时,该参数必须为此值),AddJob 的三参数 AddJob(String, String, Boolean) 重载必须在单线程单元中运行。 但是,.NET 的默认单元状态为多线程。 由于本示例假定使用非 XPSDrv 打印机,因此此默认值必须为相反值。

有两种可用于更改此默认值的方法。 一种方法是在应用程序的 Main 方法(通常为“static void Main(string[] args)”)的第一行正上方添加 STAThreadAttribute(即“[System.STAThreadAttribute()]”)即可。 但是,许多应用程序要求 Main 方法具有多线程单元状态,因此存在第二种方法:将对 AddJob(String, String, Boolean) 的调用放在单独的线程中,该线程的单元状态通过 SetApartmentState 设置为 STA。 以下示例使用第二种方法。

因此,该示例先实例化 Thread 对象,并向其传递 PrintXPS 方法,以用作 ThreadStart 参数。 (该示例的后面部分定义了 PrintXPS 方法。)接下来,将线程设置为单线程单元。 Main 方法的唯一剩余代码会启动新线程。

该示例的内容主要关于 staticBatchXPSPrinter.PrintXPS 方法。 创建打印服务器和队列后,该方法会提示用户提供包含 XPS 文件的目录。 在验证存在该目录且其中存在 *.xps 文件之后,该方法会将每个此类文件添加到打印队列。 该示例假定打印机不是 XPSDrv 打印机,因此将向 AddJob(String, String, Boolean) 方法的最后一个参数传递 false。 出于此原因,该方法先验证文件中的 XPS 标记,然后再尝试将其转换为打印机的页面描述语言。 如果验证失败,会引发异常。 该示例代码将捕获该异常,并通知用户相关信息,然后继续处理下一 XPS 文件。

class Program
{[System.MTAThreadAttribute()] // Added for clarity, but this line is redundant because MTA is the default.static void Main(string[] args){// Create the secondary thread and pass the printing method for// the constructor's ThreadStart delegate parameter. The BatchXPSPrinter// class is defined below.Thread printingThread = new Thread(BatchXPSPrinter.PrintXPS);// Set the thread that will use PrintQueue.AddJob to single threading.printingThread.SetApartmentState(ApartmentState.STA);// Start the printing thread. The method passed to the Thread// constructor will execute.printingThread.Start();}//end Main
}//end Program classpublic class BatchXPSPrinter
{public static void PrintXPS(){// Create print server and print queue.LocalPrintServer localPrintServer = new LocalPrintServer();PrintQueue defaultPrintQueue = LocalPrintServer.GetDefaultPrintQueue();// Prompt user to identify the directory, and then create the directory object.Console.Write("Enter the directory containing the XPS files: ");String directoryPath = Console.ReadLine();DirectoryInfo dir = new DirectoryInfo(directoryPath);// If the user mistyped, end the thread and return to the Main thread.if (!dir.Exists){Console.WriteLine("There is no such directory.");}else{// If there are no XPS files in the directory, end the thread// and return to the Main thread.if (dir.GetFiles("*.xps").Length == 0){Console.WriteLine("There are no XPS files in the directory.");}else{Console.WriteLine("\nJobs will now be added to the print queue.");Console.WriteLine("If the queue is not paused and the printer is working, jobs will begin printing.");// Batch process all XPS files in the directory.foreach (FileInfo f in dir.GetFiles("*.xps")){String nextFile = directoryPath + "\\" + f.Name;Console.WriteLine("Adding {0} to queue.", nextFile);try{// Print the Xps file while providing XPS validation and progress notifications.PrintSystemJobInfo xpsPrintJob = defaultPrintQueue.AddJob(f.Name, nextFile, false);}catch (PrintJobException e){Console.WriteLine("\n\t{0} could not be added to the print queue.", f.Name);if (e.InnerException.Message == "File contains corrupted data."){Console.WriteLine("\tIt is not a valid XPS file. Use the isXPS Conformance Tool to debug it.");}Console.WriteLine("\tContinuing with next XPS file.\n");}}// end for each XPS file}//end if there are no XPS files in the directory}//end if the directory does not existConsole.WriteLine("Press Enter to end program.");Console.ReadLine();}// end PrintXPS method
}// end BatchXPSPrinter class

如果使用 XPSDrv 打印机,则可将最后一个参数设置为 true。 在这种情况下,由于 XPS 是打印机的页面描述语言,该方法会将文件发送到打印机,而不会对其进行验证或将其转换为另一种页面描述语言。 如果在设计时不确定应用程序是否会使用 XPSDrv 打印机,可以修改应用程序,使其根据所发现的内容读取 IsXpsDevice 属性和分支。

由于发布 Windows Vista 和 Microsoft .NET Framework 后,最初存在几个可立即使用的 XPSDrv 打印机,可能需要将非 XPSDrv 打印机伪装为 XPSDrv 打印机。 为此,请将 Pipelineconfig.xml 添加到运行应用程序的计算机的注册表项中的以下文件列表:

HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\Print\Environments\Windows NT x86\Drivers\Version-3\<PseudoXPSPrinter>\DependentFiles

其中 <PseudoXPSPrinter> 是任一打印队列。 然后必须重新启动计算机。

此伪装允许用户将 true 传递为 AddJob(String, String, Boolean) 的最后一个参数,而不会引发异常,但由于 <PseudoXPSPrinter> 并不是真正的 XPSDrv 打印机,所以仅会打印垃圾内容。

 备注

为简单起见,以上示例通过测试是否存在 *.xps 扩展名来确定文件是否为 XPS。 但是,XPS 文件不需要具有此扩展名。 isXPS.exe(isXPS 合规性工具)是一种测试文件是否具有 XPS 有效性的方法。

这篇关于【WPF.NET开发】以编程方式打印XPS文件的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/626371

相关文章

SQL 中多表查询的常见连接方式详解

《SQL中多表查询的常见连接方式详解》本文介绍SQL中多表查询的常见连接方式,包括内连接(INNERJOIN)、左连接(LEFTJOIN)、右连接(RIGHTJOIN)、全外连接(FULLOUTER... 目录一、连接类型图表(ASCII 形式)二、前置代码(创建示例表)三、连接方式代码示例1. 内连接(I

Android里面的Service种类以及启动方式

《Android里面的Service种类以及启动方式》Android中的Service分为前台服务和后台服务,前台服务需要亮身份牌并显示通知,后台服务则有启动方式选择,包括startService和b... 目录一句话总结:一、Service 的两种类型:1. 前台服务(必须亮身份牌)2. 后台服务(偷偷干

Java深度学习库DJL实现Python的NumPy方式

《Java深度学习库DJL实现Python的NumPy方式》本文介绍了DJL库的背景和基本功能,包括NDArray的创建、数学运算、数据获取和设置等,同时,还展示了如何使用NDArray进行数据预处理... 目录1 NDArray 的背景介绍1.1 架构2 JavaDJL使用2.1 安装DJL2.2 基本操

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动

JS 实现复制到剪贴板的几种方式小结

《JS实现复制到剪贴板的几种方式小结》本文主要介绍了JS实现复制到剪贴板的几种方式小结,包括ClipboardAPI和document.execCommand这两种方法,具有一定的参考价值,感兴趣的... 目录一、Clipboard API相关属性方法二、document.execCommand优点:缺点:

Python创建Excel的4种方式小结

《Python创建Excel的4种方式小结》这篇文章主要为大家详细介绍了Python中创建Excel的4种常见方式,文中的示例代码简洁易懂,具有一定的参考价值,感兴趣的小伙伴可以学习一下... 目录库的安装代码1——pandas代码2——openpyxl代码3——xlsxwriterwww.cppcns.c

Deepseek使用指南与提问优化策略方式

《Deepseek使用指南与提问优化策略方式》本文介绍了DeepSeek语义搜索引擎的核心功能、集成方法及优化提问策略,通过自然语言处理和机器学习提供精准搜索结果,适用于智能客服、知识库检索等领域... 目录序言1. DeepSeek 概述2. DeepSeek 的集成与使用2.1 DeepSeek API

CSS弹性布局常用设置方式

《CSS弹性布局常用设置方式》文章总结了CSS布局与样式的常用属性和技巧,包括视口单位、弹性盒子布局、浮动元素、背景和边框样式、文本和阴影效果、溢出隐藏、定位以及背景渐变等,通过这些技巧,可以实现复杂... 一、单位元素vm 1vm 为视口的1%vh 视口高的1%vmin 参照长边vmax 参照长边re

JavaWeb-WebSocket浏览器服务器双向通信方式

《JavaWeb-WebSocket浏览器服务器双向通信方式》文章介绍了WebSocket协议的工作原理和应用场景,包括与HTTP的对比,接着,详细介绍了如何在Java中使用WebSocket,包括配... 目录一、概述二、入门2.1 POM依赖2.2 编写配置类2.3 编写WebSocket服务2.4 浏

配置springboot项目动静分离打包分离lib方式

《配置springboot项目动静分离打包分离lib方式》本文介绍了如何将SpringBoot工程中的静态资源和配置文件分离出来,以减少jar包大小,方便修改配置文件,通过在jar包同级目录创建co... 目录前言1、分离配置文件原理2、pom文件配置3、使用package命令打包4、总结前言默认情况下,