Двунаправленная связь с термопринтером с помощью драйверов Windows
Мне нужно разработать приложение для печати на термопринтерах, таких как протокол Epson ESC/POS или протокол ZPL Zebra. Я могу записать необработанный код на принтер, используя сетевые принтеры, принтеры с последовательным портом, а также принтеры очереди печати окна. Мои проблемы начинаются, когда мне нужно "запросить" принтер, чтобы получить его состояние (готов, ошибка, завершение бумаги и т. Д.), В этом случае мне нужно обратное соединение, я посылаю специальную команду, и принтер отвечает с некоторыми байтами состояния, С сетевыми / сокетными принтерами или принтерами с последовательным портом все понятно... с Windows-принтерами я не могу понять, как читать статус принтеров...
Есть ли решение?
ОБНОВИТЬ
Следуя предложению Ганса, я пытаюсь прочитать принтер с помощью Windows API. Это мой тест с протоколом ESC / POS.
public class WinPrinter
{
// 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);
[DllImport("winspool.Drv", EntryPoint = "ReadPrinter", SetLastError = true, CharSet = CharSet.Unicode, ExactSpelling = true, CallingConvention = CallingConvention.StdCall)]
public static extern bool ReadPrinter(IntPtr hPrinter, [MarshalAs(UnmanagedType.LPStr)] StringBuilder pBytes, Int32 dwCount, out Int32 dwNReadBytes);
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);
}
string reaDstrinG = "";
Int32 buf = 0;
Int32 pcRead;
System.Text.StringBuilder sb = new System.Text.StringBuilder();
bool read = ReadPrinter(hPrinter, sb.Append(reaDstrinG), buf, out pcRead);
ClosePrinter(hPrinter);
}
// If you did not succeed, GetLastError may give more information
// about why not.
if (bSuccess == false)
{
dwError = Marshal.GetLastWin32Error();
}
return bSuccess;
}
static void Main(string[] args)
{
String test = "\u0010\u0004\u0001";
IntPtr pBytes=Marshal.StringToCoTaskMemAnsi(test);
Int32 dwCount = test.Length;
WinPrinter.SendBytesToPrinter("BIXOLON SRP-350 (Copia 1)",pBytes,dwCount);
}
}
Я уверен, что "\u0010" (DLE EOT 1) рекомендуется запрашивать состояние принтера в протоколе ESCPOS, я протестировал его на том же принтере, подключенном через последовательный порт, и получил байт состояния.
Однако с кодом выше я всегда получаю false из метода ReadPrinter. Я проверил, что у порта принтера включена опция "двунаправленный", и я также попробовал без опции спулера.