使用命令将 BMP 文件打印到打印机

Print a BMP file to Printer using command

我需要使用命令将 BMP 文件打印到 USB 打印机。

C++ 签名是

USB_API BOOL Usb_WritePort(BOOL bUseBulkEndp, LPCVOID lpBuffer, DWORD dwNumberOfBytesToWrite, 

LPDWORD lpNumberOfBytesWritten, LPOVERLAPPED lpOverlapped);

这是我的 C# 签名

[DllImport("usbRtb.dll", CallingConvention = CallingConvention.Cdecl)]
        public static extern Boolean Usb_WritePort(bool bUseBulkEndp, IntPtr lpBuffer, UInt32 nNumberOfBytesToWrite,
          out UInt32 lpNumberOfBytesWritten, [In] ref NativeOverlapped lpOverlapped);

我试过了

 update = Usb_UpdateConnection(0, ref out1, ref out2);
        char[] files = System.Text.Encoding.ASCII.GetString(File.ReadAllBytes(@"D:\mailman.BMP")).ToCharArray();
        update = Usb_UpdateConnection(0, ref out1, ref out2);
        byte[] pr = Encoding.BigEndianUnicode.GetBytes(files);
        intpt = Marshal.StringToCoTaskMemAnsi("\x1F\x42\x4D\x50" + ASCIIEncoding.ASCII.GetString(pr));
        count = (UInt32)pr.Count() + 4;
        var read = Usb_WritePort(true, intpt, count, out written, ref natOverlap0);

附加打印机手册BMP图像打印命令部分:

C++ SDK 文档: Usb_WritePort

语法:

BOOL Usb_WritePort (BOOL                    bUseBulkEndp,
                                     LPCVOID              lpBuffer, 
                                     DWORD                dwNumberOfBytesToWrite, 
                                     LPDWORD            lpNumberOfBytesWritten, 
                                     LPOVERLAPPED lpOverlapped);

用途: 通过指定的输出端点将数据写入当前选定的设备。 需要首先使用 Usb_UpdateConnection() API 建立连接。 操作类似于在具有重叠结构的同步和异步模式下使用 WriteFile()。

参数:

bUseBulkEndp [in] 将参数设置为 TRUE 以通过 BULK OUT 端点写入数据,设置为 FALSE 以通过 INTERRUPT OUT 端点写入数据

lpBuffer [in] 指向包含要写入设备的数据的缓冲区的指针。

dwNumberOfBytesToWrite [输入] 写入设备的字节数。此数字必须小于或等于数据缓冲区的大小(以字节为单位)

lpNumberOfBytesWritten [输出] 指向接收有效写入设备的字节数的变量的指针。

lpOverlapped [in/out] 指向 OVERLAPPED 结构的可选指针,用于异步操作。如果指定此参数,Usb_WritePort 立即 returns,并在操作完成时发出事件信号。

Return值: 如果写操作成功,则为 TRUE,否则为 FALSE。如果为 FALSE,用户可以调用 GetLastError() 获取更多信息

打印机不打印任何东西。 请帮我解决这个问题。

    Marshal.StringToCoTaskMemAnsi("\x1F\x42\x4D\x50" + ASCIIEncoding.ASCII.GetString(pr));

您的代码不太正确,使用 ASCIIEncoding 是错误的。位图包含不能用 ASCII 表示的字节值。您需要做的第一件事是更改 pinvoke 声明,第二个参数需要是 byte[] 并且您最好避免重叠 I/O,因为这需要更多的 pinvoke。所以:

    [DllImport("usbRtb.dll", CallingConvention = CallingConvention.Cdecl)]
    public static extern Boolean Usb_WritePort(
        bool bUseBulkEndp, 
        byte[] lpBuffer, 
        int nNumberOfBytesToWrite,
        out int lpNumberOfBytesWritten, 
        IntPtr lpOverlapped
    );

接下来您需要做的是密切注意实际的位图。手册要求这是一个单色位图,现在不太容易得到。但是您仍然可以使用 MSPaint 创建一个,从非常小的东西开始并使用文件 + 另存为,将 "Save as type" 设置更改为 "Monochrome Bitmap (*.bmp, *.dib)"。

接下来您需要在调用该函数之前创建正确的 byte[]。它需要像这样:

    using (var fs = new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.Read)) {
        byte[] inpt = new byte[fs.Length + 1];
        inpt[0] = 0x1f;
        fs.Read(inpt, 1, (int)fs.Length);
        int written;
        bool ok = Usb_WritePort(true, inpt, inpt.Length, out written, IntPtr.Zero);
        if (!ok || written != inpt.Length) throw new Exception("USB write failed");
    }