2009-02-12 14 views
6

Sposób drukowania bezpośrednio na drukarce igłowej w języku C# przy użyciu pliku LPT1.Drukowanie na LPT1 w języku C#

Zrobiłem to na C++ z fopen, ale nie wiem jak to zrobić w języku C#.

bardzo dziękuję

+0

Mam odpowiedź, ale nie nazywam się Michael – geofftnz

+0

Domyślam się, że skopiował ją z e-maila wysłanego do kogoś zadającego to samo pytanie, po prostu zapomniałem wyjąć jego imię. – M4dRefluX

+0

Geeze ... drukarki igłowe są nadal w użyciu? Pamiętam programowanie do roku 1992, ale nie widziałem go od lat. – abelenky

Odpowiedz

14

Konfigurowanie drukarki w systemie Windows jako Generic/Text Only, a następnie na niej drukować.

Oto kod, którego używam do drukowania na drukarce etykiet z własnym zakodowanym formatem tekstu.

using System; 
using System.Collections.Generic; 
using System.Text; 
using System.Drawing; 
using System.Drawing.Printing; 
using System.Windows.Forms; 
using System.Runtime.InteropServices; 
using System.IO; 

namespace whatever { 

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. 
     bool success = SendBytesToPrinter(szPrinterName, pBytes, dwCount); 
     Marshal.FreeCoTaskMem(pBytes); 
     return success; 
    } 
} 
} 

Wtedy nazywa się to z innej klasy tak:

private bool PrintLabels(string printerName) 
    { 
     return RawPrinterHelper.SendStringToPrinter(printerName, this.Text); 
    } 

(edit: jeśli ktoś widzi żadnych poważnych fubars związanych niezagospodarowanych zasobów, daj mi znać w komentarzach)

+0

Człowieku, Michael jest niesamowity. –

+0

Świetna robota, Michael! Co powiesz na randkę? ; P – strager

+0

brb zmieniając nazwisko na Michaela ... – geofftnz

1

Może nie można drukować na urządzeniu "LPT1", ale na "głupiej drukarce teletropowej", którą trzeba najpierw zainstalować. Jeśli istnieją w nowoczesnych oknach ... Powinieneś więc drukować na "mydumbprinter", który z kolei używa LPT1 jako portu drukarki. Wydało mi się, że podążyłem za linkiem źródłowym autorstwa Cocowalla powyżej - tam wybieramy drukarkę w PrinterDialog. Wynikowa nazwa drukarki byłaby bardziej "Laserjet" niż "LPT1".

0

W języku C# jest to możliwe, najpierw należy połączyć się z tym portem za pomocą metody CreateFile, a następnie otworzyć strumień plików do tego portu, aby w końcu do niego napisać. Oto przykładowa klasa, która zapisuje dwie linie do drukarki na LPT1.

using Microsoft.Win32.SafeHandles; 
using System; 
using System.IO; 
using System.Runtime.InteropServices; 

namespace YourNamespace 
{ 
public static class Print2LPT 
     { 
      [DllImport("kernel32.dll", SetLastError = true)] 
      static extern SafeFileHandle CreateFile(string lpFileName, FileAccess dwDesiredAccess,uint dwShareMode, IntPtr lpSecurityAttributes, FileMode dwCreationDisposition, uint dwFlagsAndAttributes, IntPtr hTemplateFile); 

      public static bool Print() 
      { 
       string nl = Convert.ToChar(13).ToString() + Convert.ToChar(10).ToString(); 
       bool IsConnected= false; 

       string sampleText ="Hello World!" + nl + 
       "Enjoy Printing...";  
       try 
       { 
        Byte[] buffer = new byte[sampleText.Length]; 
        buffer = System.Text.Encoding.ASCII.GetBytes(sampleText); 

        SafeFileHandle fh = CreateFile("LPT1:", FileAccess.Write, 0, IntPtr.Zero, FileMode.OpenOrCreate, 0, IntPtr.Zero); 
        if (!fh.IsInvalid) 
        { 
         IsConnected= true;      
         FileStream lpt1 = new FileStream(fh,FileAccess.ReadWrite); 
         lpt1.Write(buffer, 0, buffer.Length); 
         lpt1.Close(); 
        } 

       } 
       catch (Exception ex) 
       { 
        string message = ex.Message; 
       } 

       return IsConnected; 
      } 
     } 
} 

Zakładając, że drukarka jest podłączona przez port w LPT1, jeśli nie trzeba będzie dostosować metodę CreateFile pasujące do portu, z którego korzystasz.

można wywołać metodę w dowolnym miejscu w programie z poniższej linii

Print2LPT.Print(); 

myślę, że to jest najkrótsza i najbardziej wydajne rozwiązanie problemu.