web-dev-qa-db-ja.com

USB経由で生のZPLをZebraプリンターに送信する

通常、Zebra LP 2844-ZをUSBポートに接続すると、コンピューターはそれをプリンターとして認識し、他の一般的なプリンターと同様にメモ帳から印刷できます。ただし、私のアプリケーションにはバーコード機能がいくつかあります。私のアプリケーションは入力を解析し、ZPLのメモリ内文字列を生成します。このZPLデータをUSBデバイスに送信するにはどうすればよいですか?

21

COMポートを介してZebraプリンターに書き込む簡単な方法を見つけました。 Windowsコントロールパネルに移動し、新しいプリンターを追加しました。ポートには、COM1(プリンターが接続されているポート)を選択しました。 「Generic/Text Only」プリンタードライバーを使用しました。印刷スプーラー(プリンター設定の標準オプション)とすべての高度な印刷オプションを無効にしました。これで、任意の文字列をそのプリンターに印刷することができ、文字列にZPLが含まれている場合、プリンターはZPLを正常にレンダリングします!特別な「開始シーケンス」やそのようなファンキーなものは必要ありません。簡単にするためにイェーイ!

15

私は答えを見つけました...または少なくとも、最も簡単な答え(複数ある場合)。プリンターをインストールしたときに、名前を「ICS Label Printer」に変更しました。パススルーZPLコマンドを許可するオプションを変更する方法は次のとおりです。

  1. 「ICS Label Printer」を右クリックして、「プロパティ」を選択します。
  2. [全般]タブで、[印刷設定...]ボタンをクリックします。
  3. [詳細設定]タブで、[その他]ボタンをクリックします。
  4. 「パススルーモードを有効にする」というラベルの付いたボックスにチェックがあることを確認します。
  5. 「開始シーケンス:」が「$ {」であることを確認してください。
  6. 「終了シーケンス:」が「} $」であることを確認します。
  7. 「閉じる」ボタンをクリックします。
  8. 「OK」ボタンをクリックします。
  9. 「OK」ボタンをクリックします。

私のコードでは、ZPLの先頭に「$ {」を追加し、末尾に「} $」を追加して、プレーンテキストとして印刷するだけです。これは、「ZDesigner LP 2844-Zプリンターバージョン2.6.42(ビルド2382)用のWindowsドライバー」にあります。チャームのように機能します!

17

Visual Studio C#ソリューションhttp://support.Microsoft.com/kb/322091 にあります)

ステップ1。)クラスRawPrinterHelperを作成...

using System;
using System.IO;
using System.Runtime.InteropServices;

public class RawPrinterHelper
{
    // Structure and API declarions:
    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
    public 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)]
    public 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)]
    public static extern bool ClosePrinter(IntPtr hPrinter);

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

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

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

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

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

    // SendBytesToPrinter()
    // When the function is given a printer name and an unmanaged array
    // of bytes, the function sends those bytes to the print queue.
    // Returns true on success, false on failure.
    public static bool SendBytesToPrinter(string szPrinterName, IntPtr pBytes, Int32 dwCount)
    {
        Int32 dwError = 0, dwWritten = 0;
        IntPtr hPrinter = new IntPtr(0);
        DOCINFOA di = new DOCINFOA();
        bool bSuccess = false; // Assume failure unless you specifically succeed.

        di.pDocName = "My C#.NET RAW Document";
        di.pDataType = "RAW";

        // Open the printer.
        if (OpenPrinter(szPrinterName.Normalize(), out hPrinter, IntPtr.Zero))
        {
            // Start a document.
            if (StartDocPrinter(hPrinter, 1, di))
            {
                // Start a page.
                if (StartPagePrinter(hPrinter))
                {
                    // Write your bytes.
                    bSuccess = WritePrinter(hPrinter, pBytes, dwCount, out dwWritten);
                    EndPagePrinter(hPrinter);
                }
                EndDocPrinter(hPrinter);
            }
            ClosePrinter(hPrinter);
        }
        // If you did not succeed, GetLastError may give more information
        // about why not.
        if (bSuccess == false)
        {
            dwError = Marshal.GetLastWin32Error();
        }
        return bSuccess;
    }

    public static bool SendFileToPrinter(string szPrinterName, string szFileName)
    {
        // Open the file.
        FileStream fs = new FileStream(szFileName, FileMode.Open);
        // Create a BinaryReader on the file.
        BinaryReader br = new BinaryReader(fs);
        // Dim an array of bytes big enough to hold the file's contents.
        Byte[] bytes = new Byte[fs.Length];
        bool bSuccess = false;
        // Your unmanaged pointer.
        IntPtr pUnmanagedBytes = new IntPtr(0);
        int nLength;

        nLength = Convert.ToInt32(fs.Length);
        // Read the contents of the file into the array.
        bytes = br.ReadBytes(nLength);
        // Allocate some unmanaged memory for those bytes.
        pUnmanagedBytes = Marshal.AllocCoTaskMem(nLength);
        // Copy the managed byte array into the unmanaged array.
        Marshal.Copy(bytes, 0, pUnmanagedBytes, nLength);
        // Send the unmanaged bytes to the printer.
        bSuccess = SendBytesToPrinter(szPrinterName, pUnmanagedBytes, nLength);
        // Free the unmanaged memory that you allocated earlier.
        Marshal.FreeCoTaskMem(pUnmanagedBytes);
        return bSuccess;
    }
    public static bool SendStringToPrinter(string szPrinterName, string szString)
    {
        IntPtr pBytes;
        Int32 dwCount;
        // How many characters are in the string?
        dwCount = szString.Length;
        // Assume that the printer is expecting ANSI text, and then convert
        // the string to ANSI text.
        pBytes = Marshal.StringToCoTaskMemAnsi(szString);
        // Send the converted ANSI string to the printer.
        SendBytesToPrinter(szPrinterName, pBytes, dwCount);
        Marshal.FreeCoTaskMem(pBytes);
        return true;
    }
}

ステップ2。)テキストボックスとボタンを含むフォームを作成します(この例では、送信するZPLがテキストボックスに保持されます)。ボタンクリックイベントでコードを追加...

private void button1_Click(object sender, EventArgs e)
        {
            // Allow the user to select a printer.
            PrintDialog pd = new PrintDialog();
            pd.PrinterSettings = new PrinterSettings();
            if (DialogResult.OK == pd.ShowDialog(this))
            {
                // Send a printer-specific to the printer.
                RawPrinterHelper.SendStringToPrinter(pd.PrinterSettings.PrinterName, textBox1.Text);
                MessageBox.Show("Data sent to printer.");
            }
            else
            {
                MessageBox.Show("Data not sent to printer.");
            }
        }

このソリューションを使用すると、特定の要件を満たすように微調整できます。おそらく、特定のプリンターをハードコーディングします。おそらく、テキストボックスからではなく、動的にZPLテキストを派生させます。なんでも。おそらく、グラフィカルインターフェイスは必要ありませんが、これはZPLの送信方法を示しています。用途はニーズによって異なります。

11
barrypicker

あなたは言語に言及していないので、CのまっすぐなWindows APIでそれを行う方法について、いくつかのヒントを紹介します。

まず、 OpenPrinter を使用してプリンターへの接続を開きます。次に、 StartDocPrinterDOC_INFO_1pDatatypeフィールドを持つドキュメントを開始します構造が"RAW"に設定されている-これは、プリンタードライバーに、プリンターに送られるものをエンコードせず、そのまま渡すように指示します。 StartPagePrinterを使用して最初のページを示し、WritePrinterを使用してデータをプリンターに送信し、完了したらEndPagePrinterEndDocPrinterおよびClosePrinterを使用して閉じます。

8
Mark Ransom

ZPLは正しい方法です。ほとんどの場合、GDIコマンドに抽象化するドライバーを使用するのが正しいです。ただし、Zebraラベルプリンターは特殊なケースです。Zebraプリンターに印刷する最良の方法は、ZPLを直接生成することです。 Zebraプリンターの実際のプリンタードライバーは「プレーンテキスト」プリンターです-ほとんどのプリンターにドライバーがあると考える意味で更新または変更できる「ドライバー」はありません。 。

3
Thaeli

プリンターに共有をインストールします。\ localhost\zebra ZPLをテキストとして送信し、最初にコピーを試してください:

file.zpl\localhost\zebraをコピーします

非常にシンプル、ほとんどコーディングなし。

0
Thomas

そのために8時間を費やしました。それは単純だ...

あなたはそのようなコードを持っている必要があります:

private const int GENERIC_WRITE = 0x40000000;

//private const int OPEN_EXISTING = 3;
private const int OPEN_EXISTING = 1;
private const int FILE_SHARE_WRITE = 0x2;
private StreamWriter _fileWriter;
private FileStream _outFile;
private int _hPort;

変数の内容を3(開いているファイルが既に存在する)から1(新しいファイルを作成する)に変更します。 Windows 7およびXPで動作します。

0
Rodrigo Aquino