gpt4 book ai didi

c# - 将 FixedDocument/XPS 打印为 PDF 而不显示文件保存对话框

转载 作者:行者123 更新时间:2023-11-30 15:50:57 28 4
gpt4 key购买 nike

我有一个 FixedDocument,我允许用户在 WPF GUI 中预览,然后打印到纸上而不显示任何 Windows 打印对话框,如下所示:

private void Print()
{
PrintQueueCollection printQueues;
using (var printServer = new PrintServer())
{
var flags = new[] { EnumeratedPrintQueueTypes.Local };
printQueues = printServer.GetPrintQueues(flags);
}

//SelectedPrinter.FullName can be something like "Microsoft Print to PDF"
var selectedQueue = printQueues.SingleOrDefault(pq => pq.FullName == SelectedPrinter.FullName);

if (selectedQueue != null)
{
var myTicket = new PrintTicket
{
CopyCount = 1,
PageOrientation = PageOrientation.Portrait,
OutputColor = OutputColor.Color,
PageMediaSize = new PageMediaSize(PageMediaSizeName.ISOA4)
};

var mergeTicketResult = selectedQueue.MergeAndValidatePrintTicket(selectedQueue.DefaultPrintTicket, myTicket);
var printTicket = mergeTicketResult.ValidatedPrintTicket;

// TODO: Make sure merge was OK

// Calling GetPrintCapabilities with our ticket allows us to use
// the OrientedPageMediaHeight/OrientedPageMediaWidth properties
// and the PageImageableArea property to calculate the minimum
// document margins supported by the printer. Very important!
var printCapabilities = queue.GetPrintCapabilities(myTicket);
var fixedDocument = GenerateFixedDocument(printCapabilities);

var dlg = new PrintDialog
{
PrintTicket = printTicket,
PrintQueue = selectedQueue
};

dlg.PrintDocument(fixedDocument.DocumentPaginator, "test document");
}
}

问题是我还想通过提供文件目标路径而不显示任何 Windows 对话框来支持虚拟/文件打印机,即 PDF 打印,但这似乎不适用于 PrintDialog.

我真的很想尽可能避免使用第 3 方库,所以至少现在,我不想使用 PdfSharp 之类的工具将 XPS 转换为 PDF。 更正:似乎从最新版本的 PdfSharp 中删除了 XPS 转换支持。

经过一些研究,似乎直接打印到文件的唯一方法是使用 PrintDocument,其中可以设置 PrintFileNamePrintToFilePrinterSettings 对象中,但是没有办法给出实际的文档内容,而是我们需要订阅 PrintPage 事件并做一些 System .Drawing.Graphics 创建文档的操作。

这是我试过的代码:

var printDoc = new PrintDocument
{
PrinterSettings =
{
PrinterName = SelectedPrinter.FullName,
PrintFileName = destinationFilePath,
PrintToFile = true
},
PrintController = new StandardPrintController()
};

printDoc.PrintPage += OnPrintPage; // Without this line, we get a blank PDF
printDoc.Print();

然后是 PrintPage 的处理程序,我们需要在其中构建文档:

private void OnPrintPage(object sender, PrintPageEventArgs e)
{
// What to do here?
}

我认为其他可行的方法是使用 System.Windows.Forms.PrintDialog 类,但它也需要一个 PrintDocument。我能够像这样轻松地创建 XPS 文件:

var pkg = Package.Open(destinationFilePath, FileMode.Create);
var doc = new XpsDocument(pkg);
var writer = XpsDocument.CreateXpsDocumentWriter(doc);
writer.Write(PreviewDocument.DocumentPaginator);
pkg.Flush();
pkg.Close();

但它不是 PDF,如果没有第 3 方库,似乎无法将其转换为 PDF。

是否有可能做一个 hack,自动填充文件名,然后单击 PrintDialog 上的保存?

谢谢!

编辑:可以使用 Microsoft.Office.Interop.Word 将 Word 文档直接打印为 PDF,但似乎没有从 XPS 转换的简单方法/FixedDocument 到 Word。

编辑: 到目前为止,最好的方法似乎是获取 PdfSharp 1.31 中存在的旧 XPS 到 PDF 转换代码。我抓取了源代码并构建了它,导入了 DLL,它就可以工作了。感谢 Nathan Jones,查看他关于此的博客文章 here .

最佳答案

解决了!谷歌搜索后,我受到了直接调用 Windows 打印机的 P/Invoke 方法的启发。

所以解决方案是使用 Print Spooler API直接调用 Windows 中可用的 Microsoft Print to PDF 打印机的函数(确保已安装该功能!)并为 WritePrinter 函数提供 XPS 文件的字节。

我相信这是可行的,因为 Microsoft PDF 打印机驱动程序理解 XPS 页面描述语言。这可以通过检查打印队列的 IsXpsDevice 属性来检查。

必须在 Windows 中安装“Microsoft Print to PDF”功能才能使用!

代码如下:

using System;
using System.Linq;
using System.Printing;
using System.Runtime.InteropServices;

public static class PdfFilePrinter
{
private const string PdfPrinterDriveName = "Microsoft Print To PDF";

[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
private class DOCINFOA
{
[MarshalAs(UnmanagedType.LPStr)]
public string pDocName;
[MarshalAs(UnmanagedType.LPStr)]
public string pOutputFile;
[MarshalAs(UnmanagedType.LPStr)]
public string pDataType;
}

[DllImport("winspool.drv", EntryPoint = "OpenPrinterA", SetLastError = true, CharSet = CharSet.Ansi, ExactSpelling = true, CallingConvention = CallingConvention.StdCall)]
private static extern bool OpenPrinter([MarshalAs(UnmanagedType.LPStr)] string szPrinter, out IntPtr hPrinter, IntPtr pd);

[DllImport("winspool.drv", EntryPoint = "ClosePrinter", SetLastError = true, ExactSpelling = true, CallingConvention = CallingConvention.StdCall)]
private static extern bool ClosePrinter(IntPtr hPrinter);

[DllImport("winspool.drv", EntryPoint = "StartDocPrinterA", SetLastError = true, CharSet = CharSet.Ansi, ExactSpelling = true, CallingConvention = CallingConvention.StdCall)]
private static extern int StartDocPrinter(IntPtr hPrinter, int level, [In, MarshalAs(UnmanagedType.LPStruct)] DOCINFOA di);

[DllImport("winspool.drv", EntryPoint = "EndDocPrinter", SetLastError = true, ExactSpelling = true, CallingConvention = CallingConvention.StdCall)]
private static extern bool EndDocPrinter(IntPtr hPrinter);

[DllImport("winspool.drv", EntryPoint = "StartPagePrinter", SetLastError = true, ExactSpelling = true, CallingConvention = CallingConvention.StdCall)]
private static extern bool StartPagePrinter(IntPtr hPrinter);

[DllImport("winspool.drv", EntryPoint = "EndPagePrinter", SetLastError = true, ExactSpelling = true, CallingConvention = CallingConvention.StdCall)]
private static extern bool EndPagePrinter(IntPtr hPrinter);

[DllImport("winspool.drv", EntryPoint = "WritePrinter", SetLastError = true, ExactSpelling = true, CallingConvention = CallingConvention.StdCall)]
private static extern bool WritePrinter(IntPtr hPrinter, IntPtr pBytes, int dwCount, out int dwWritten);

public static void PrintXpsToPdf(byte[] bytes, string outputFilePath, string documentTitle)
{
// Get Microsoft Print to PDF print queue
var pdfPrintQueue = GetMicrosoftPdfPrintQueue();

// Copy byte array to unmanaged pointer
var ptrUnmanagedBytes = Marshal.AllocCoTaskMem(bytes.Length);
Marshal.Copy(bytes, 0, ptrUnmanagedBytes, bytes.Length);

// Prepare document info
var di = new DOCINFOA
{
pDocName = documentTitle,
pOutputFile = outputFilePath,
pDataType = "RAW"
};

// Print to PDF
var errorCode = SendBytesToPrinter(pdfPrintQueue.Name, ptrUnmanagedBytes, bytes.Length, di, out var jobId);

// Free unmanaged memory
Marshal.FreeCoTaskMem(ptrUnmanagedBytes);

// Check if job in error state (for example not enough disk space)
var jobFailed = false;
try
{
var pdfPrintJob = pdfPrintQueue.GetJob(jobId);
if (pdfPrintJob.IsInError)
{
jobFailed = true;
pdfPrintJob.Cancel();
}
}
catch
{
// If job succeeds, GetJob will throw an exception. Ignore it.
}
finally
{
pdfPrintQueue.Dispose();
}

if (errorCode > 0 || jobFailed)
{
try
{
if (File.Exists(outputFilePath))
{
File.Delete(outputFilePath);
}
}
catch
{
// ignored
}
}

if (errorCode > 0)
{
throw new Exception($"Printing to PDF failed. Error code: {errorCode}.");
}

if (jobFailed)
{
throw new Exception("PDF Print job failed.");
}
}

private static int SendBytesToPrinter(string szPrinterName, IntPtr pBytes, int dwCount, DOCINFOA documentInfo, out int jobId)
{
jobId = 0;
var dwWritten = 0;
var success = false;

if (OpenPrinter(szPrinterName.Normalize(), out var hPrinter, IntPtr.Zero))
{
jobId = StartDocPrinter(hPrinter, 1, documentInfo);
if (jobId > 0)
{
if (StartPagePrinter(hPrinter))
{
success = WritePrinter(hPrinter, pBytes, dwCount, out dwWritten);
EndPagePrinter(hPrinter);
}

EndDocPrinter(hPrinter);
}

ClosePrinter(hPrinter);
}

// TODO: The other methods such as OpenPrinter also have return values. Check those?

if (success == false)
{
return Marshal.GetLastWin32Error();
}

return 0;
}

private static PrintQueue GetMicrosoftPdfPrintQueue()
{
PrintQueue pdfPrintQueue = null;

try
{
using (var printServer = new PrintServer())
{
var flags = new[] { EnumeratedPrintQueueTypes.Local };
// FirstOrDefault because it's possible for there to be multiple PDF printers with the same driver name (though unusual)
// To get a specific printer, search by FullName property instead (note that in Windows, queue name can be changed)
pdfPrintQueue = printServer.GetPrintQueues(flags).FirstOrDefault(lq => lq.QueueDriver.Name == PdfPrinterDriveName);
}

if (pdfPrintQueue == null)
{
throw new Exception($"Could not find printer with driver name: {PdfPrinterDriveName}");
}

if (!pdfPrintQueue.IsXpsDevice)
{
throw new Exception($"PrintQueue '{pdfPrintQueue.Name}' does not understand XPS page description language.");
}

return pdfPrintQueue;
}
catch
{
pdfPrintQueue?.Dispose();
throw;
}
}
}

用法:

public static void FixedDocument2Pdf(FixedDocument fd)
{
// Convert FixedDocument to XPS file in memory
var ms = new MemoryStream();
var package = Package.Open(ms, FileMode.Create);
var doc = new XpsDocument(package);
var writer = XpsDocument.CreateXpsDocumentWriter(doc);
writer.Write(fd.DocumentPaginator);
doc.Close();
package.Close();

// Get XPS file bytes
var bytes = ms.ToArray();
ms.Dispose();

// Print to PDF
var outputFilePath = @"C:\tmp\test.pdf";
PdfFilePrinter.PrintXpsToPdf(bytes, outputFilePath, "Document Title");
}

在上面的代码中,我没有直接给出打印机名称,而是通过使用驱动程序名称查找打印队列来获取名称,因为我相信它是常量,而打印机名称实际上可以在 Windows 中更改,我也不知道它是否受到本地化的影响,所以这种方式更安全。

注意:在开始打印操作之前检查可用磁盘空间大小是个好主意,因为我找不到可靠的方法来查明错误是否是磁盘空间不足。一种想法是将 XPS 字节数组长度乘以一个像 3 这样的魔数(Magic Number),然后检查磁盘上是否有那么多空间。此外,提供一个空字节数组或带有伪造数据的字节数组不会在任何地方失败,但会产生损坏的 PDF 文件。

评论注释:仅使用 FileStream 读取 XPS 文件是行不通的。我们必须从内存中的 Package 创建一个 XpsDocument,然后像这样从 MemomryStream 中读取字节:

public static void PrintFile(string xpsSourcePath, string pdfOutputPath)
{
// Write XPS file to memory stream
var ms = new MemoryStream();
var package = Package.Open(ms, FileMode.Create);
var doc = new XpsDocument(package);
var writer = XpsDocument.CreateXpsDocumentWriter(doc);
writer.Write(xpsSourcePath);
doc.Close();
package.Close();

// Get XPS file bytes
var bytes = ms.ToArray();
ms.Dispose();

// Print to PDF
PdfPrinter.PrintXpsToPdf(bytes, pdfOutputPath, "Document title");
}

关于c# - 将 FixedDocument/XPS 打印为 PDF 而不显示文件保存对话框,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58517394/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com