11 Stimmen

Xps-Druck vom Windows-Dienst

Ich versuche, XPS-Dokumente von einem Windows-Dienst auf dem .net-Framework zu drucken. Da Microsoft das Drucken weder mit System.Drawing.Printing noch mit System.Printing (WPF) unterstützt, verwende ich die native XPSPrint API. Dies wird mir von Aspose vorgeschlagen in http://www.aspose.com/documentation/.net-components/aspose.words-for-.net/howto-print-a-document-on-a-server-via-the-xpsprint-api.html .

Wenn ich versuche, ein XPS-Dokument von einem Windows-Dienst aus zu drucken, enthält das Ergebnis seltsame Zeichen anstelle des gewünschten Textes.

Ich habe es mit verschiedenen Druckern (einschließlich virtueller Drucker wie z.B. PDFCreator), verschiedenen Benutzern und Benutzerrechten für den Dienst, verschiedenen xps-Generatoren (Aspose, Word 2007, Word 2010), verschiedenen Plattformen (Windows 7, Windows 2008 R2) versucht, aber alle haben das gleiche Ergebnis.

Weiß jemand, wie man dieses Problem lösen kann? Für jede Hilfe wären wir dankbar!

Für diejenigen, die es ausprobieren möchten, habe ich einige Dateien zur Verfügung gestellt:

https://docs.google.com/leaf?id=0B4J93Ly5WzQKNWU2ZjM0MDYtMjFiMi00NzM0LTg4MTgtYjVlNDA5NWQyMTc3&hl=nl

  • document.xps: das zu druckende XPS-Dokument
  • document_printed_to_pdfcreator.pdf: das gedruckte Dokument, das zeigt, was schief läuft
  • XpsPrintTest.zip: eine VS2010-Beispiellösung mit dem Beispielcode

Der Beispielcode für den verwalteten Windows-Dienst lautet:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Diagnostics;
using System.Linq;
using System.ServiceProcess;
using System.Text;
using System.IO;
using System.Threading;
using System.Runtime.InteropServices;

namespace PrintXpsService
{
public partial class XpsPrintService : ServiceBase
{
    // Change name of printer here
    private String f_printerName = "PDFCreator";

    // path to some file where logging is done
    private String f_logFile = @"C:\temp\testdoc\xps_printing_service_log.txt";

    // path to xps file to print
    private String f_xpsFile = @"C:\temp\testdoc\document.xps";

    public XpsPrintService()
    {
        InitializeComponent();
    }

    private void Log(String fmt, params Object[] args)
    {
        try
        {
            DateTime now = DateTime.Now;

            using (StreamWriter wrt = new StreamWriter(f_logFile, true))
            {
                wrt.Write("{0} {1} - ", now.ToShortDateString(), now.ToShortTimeString());
                wrt.WriteLine(fmt, args);
            }
        }
        catch (Exception ex)
        {
        }
    }

    protected override void OnStart(string[] args)
    {
        // uncomment to allow to connect debugger
        //int i = 0;
        //while (i == 0)
        //{
        //    if (i == 0)
        //    {
        //        Thread.Sleep(1000);
        //    }
        //}

        Log("Starting Service");
        try
        {
            Log("Printing xps file {0}", f_xpsFile);

            using (Stream stream = new FileStream(f_xpsFile, FileMode.Open, FileAccess.Read))
            {
                Log("Starting to print on printer {0}", f_printerName);
                String jobName = f_xpsFile;
                this.Print(stream, jobName);
            }
            Log("Document printed");
        }
        catch (Exception ex)
        {
            Log("Exception during execution: {0}", ex.Message);
            Log("  {0}", ex.StackTrace);
            Exception inner = ex.InnerException;
            while (inner != null)
            {
                Log("=== Inner Exception: {0}", inner.Message);
                Log("    {0}", inner.StackTrace);
                inner = inner.InnerException;
            }
        }
    }

    protected override void OnStop()
    {
    }

    public void Print(Stream stream, String jobName)
    {
        String printerName = f_printerName;
        IntPtr completionEvent = CreateEvent(IntPtr.Zero, true, false, null);
        try
        {
            IXpsPrintJob job;
            IXpsPrintJobStream jobStream;

            StartJob(printerName, jobName, completionEvent, out job, out jobStream);
            CopyJob(stream, job, jobStream);
            WaitForJob(completionEvent, -1);
            CheckJobStatus(job);
        }
        finally
        {
            if (completionEvent != IntPtr.Zero)
                CloseHandle(completionEvent);
        }
    }

    private void StartJob(String printerName,
        String jobName, IntPtr completionEvent,
        out IXpsPrintJob job,
        out IXpsPrintJobStream jobStream)
    {
        int result = StartXpsPrintJob(printerName, jobName, null, IntPtr.Zero, completionEvent,
            null, 0, out job, out jobStream, IntPtr.Zero);
        if (result != 0)
            throw new Win32Exception(result);
    }

    private void CopyJob(Stream stream, IXpsPrintJob job, IXpsPrintJobStream jobStream)
    {
        try
        {
            byte[] buff = new byte[4096];
            while (true)
            {
                uint read = (uint)stream.Read(buff, 0, buff.Length);
                if (read == 0)
                    break;
                uint written;
                jobStream.Write(buff, read, out written);

                if (read != written)
                    throw new Exception("Failed to copy data to the print job stream.");
            }

            // Indicate that the entire document has been copied.
            jobStream.Close();
        }
        catch (Exception)
        {
            // Cancel the job if we had any trouble submitting it.
            job.Cancel();
            throw;
        }
    }

    private void WaitForJob(IntPtr completionEvent, int timeout)
    {
        if (timeout < 0)
            timeout = -1;

        switch (WaitForSingleObject(completionEvent, timeout))
        {
            case WAIT_RESULT.WAIT_OBJECT_0:
                // Expected result, do nothing.
                break;

            case WAIT_RESULT.WAIT_TIMEOUT:
                // timeout expired
                throw new Exception("Timeout expired");

            case WAIT_RESULT.WAIT_FAILED:
                throw new Exception("Wait for the job to complete failed");

            default:
                throw new Exception("Unexpected result when waiting for the print job.");
        }
    }

    private void CheckJobStatus(IXpsPrintJob job)
    {
        XPS_JOB_STATUS jobStatus;
        job.GetJobStatus(out jobStatus);
        switch (jobStatus.completion)
        {
            case XPS_JOB_COMPLETION.XPS_JOB_COMPLETED:
                // Expected result, do nothing.
                break;
            case XPS_JOB_COMPLETION.XPS_JOB_IN_PROGRESS:
                // expected, do nothing, can occur when printer is paused
                break;
            case XPS_JOB_COMPLETION.XPS_JOB_FAILED:
                throw new Win32Exception(jobStatus.jobStatus);
            default:
                throw new Exception("Unexpected print job status.");
        }
    }

    [DllImport("XpsPrint.dll", EntryPoint = "StartXpsPrintJob")]
    private static extern int StartXpsPrintJob(
        [MarshalAs(UnmanagedType.LPWStr)] String printerName,
        [MarshalAs(UnmanagedType.LPWStr)] String jobName,
        [MarshalAs(UnmanagedType.LPWStr)] String outputFileName,
        IntPtr progressEvent,   // HANDLE
        IntPtr completionEvent, // HANDLE
        [MarshalAs(UnmanagedType.LPArray)] byte[] printablePagesOn,
        UInt32 printablePagesOnCount,
        out IXpsPrintJob xpsPrintJob,
        out IXpsPrintJobStream documentStream,
        IntPtr printTicketStream);  // This is actually "out IXpsPrintJobStream", but we don't use it and just want to pass null, hence IntPtr.

    [DllImport("Kernel32.dll", SetLastError = true)]
    private static extern IntPtr CreateEvent(IntPtr lpEventAttributes, bool bManualReset, bool bInitialState, string lpName);

    [DllImport("Kernel32.dll", SetLastError = true, ExactSpelling = true)]
    private static extern WAIT_RESULT WaitForSingleObject(IntPtr handle, Int32 milliseconds);

    [DllImport("Kernel32.dll", SetLastError = true)]
    [return: MarshalAs(UnmanagedType.Bool)]
    private static extern bool CloseHandle(IntPtr hObject);
}

/// <summary>
/// This interface definition is HACKED.
/// 
/// It appears that the IID for IXpsPrintJobStream specified in XpsPrint.h as 
/// MIDL_INTERFACE("7a77dc5f-45d6-4dff-9307-d8cb846347ca") is not correct and the RCW cannot return it.
/// But the returned object returns the parent ISequentialStream inteface successfully.
/// 
/// So the hack is that we obtain the ISequentialStream interface but work with it as 
/// with the IXpsPrintJobStream interface. 
/// </summary>
[Guid("0C733A30-2A1C-11CE-ADE5-00AA0044773D")]  // This is IID of ISequenatialSteam.
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
interface IXpsPrintJobStream
{
    // ISequentualStream methods.
    void Read([MarshalAs(UnmanagedType.LPArray)] byte[] pv, uint cb, out uint pcbRead);
    void Write([MarshalAs(UnmanagedType.LPArray)] byte[] pv, uint cb, out uint pcbWritten);
    // IXpsPrintJobStream methods.
    void Close();
}

[Guid("5ab89b06-8194-425f-ab3b-d7a96e350161")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
interface IXpsPrintJob
{
    void Cancel();
    void GetJobStatus(out XPS_JOB_STATUS jobStatus);
}

[StructLayout(LayoutKind.Sequential)]
struct XPS_JOB_STATUS
{
    public UInt32 jobId;
    public Int32 currentDocument;
    public Int32 currentPage;
    public Int32 currentPageTotal;
    public XPS_JOB_COMPLETION completion;
    public Int32 jobStatus; // UInt32
};

enum XPS_JOB_COMPLETION
{
    XPS_JOB_IN_PROGRESS = 0,
    XPS_JOB_COMPLETED = 1,
    XPS_JOB_CANCELLED = 2,
    XPS_JOB_FAILED = 3
}

enum WAIT_RESULT
{
    WAIT_OBJECT_0 = 0,
    WAIT_ABANDONED = 0x80,
    WAIT_TIMEOUT = 0x102,
    WAIT_FAILED = -1 // 0xFFFFFFFF
}
}

Hinweis: Einige Links für weitere Informationen:

CodeJaeger.com

CodeJaeger ist eine Gemeinschaft für Programmierer, die täglich Hilfe erhalten..
Wir haben viele Inhalte, und Sie können auch Ihre eigenen Fragen stellen oder die Fragen anderer Leute lösen.

Powered by:

X