斑马打印机客户端GET和POST,以及后端两种打印方式。

时间:2023-01-11 15:20:56

斑马打印机客户端GET和POST,以及后端两种打印方式。

背景环境:打印机安装在客户端外网。当用户登录时,通过ajax取服务器数据,返回打印机命令,然后客户端通过JS发送给斑马打印机。

1、使用Get方式打印

1.1 前端页面js代码

jQuery(function () {
$("#btnRePrint").click(function () {
//var cartonId = "450002241530221";
var cartonId = "";
if ($("input[name='checkbox']:checked").length == 1) {
// 1、获取重打打的数据,返回箱号
var trs = $("#data tr");
for (var i = 0; i < trs.length; i++) {
if ($("#checkbox_" + i).attr('checked') == true) {
var strVendorId = trs[i].cells[2].innerText; // 供应商
var strPoId = trs[i].cells[4].innerText; // 采购订单
var strPoSeq = trs[i].cells[5].innerText; // 采购订单行项
var strMatId = trs[i].cells[7].innerText // 物料号

$.ajax({
url: "Handler/CreatedliverGetRePrintData.ashx?VendorId=" + strVendorId + "&PoId=" + strPoId + "&PoSeq=" + strPoSeq + "&MatId=" + strMatId,
async:false,
success: function (result, status) {
debugger;
cartonId = result;
}
});
}
}

// 2、调用Web Api 返回zpl命令
//jQuery.support.cors = true;
$.ajax({
url: "Handler/CreatedeliverRePrint_Ajax.ashx?CartonId=" + cartonId,
async:false,
success: function (result, status) {
debugger;

var arrayResult = result.split('|');
// 返回消息格式:S|Zpl命令,E|错误提醒
if (arrayResult[0].toUpperCase() == "\"ZPL".toUpperCase()) {
var zpl = arrayResult[1].substring(0, arrayResult[1].length - 1);
Print(zpl);
alert("重打成功!");
}
if (arrayResult[0].toUpperCase() == "\"Tip".toUpperCase()) {
alert(arrayResult[1]);
}
if (arrayResult[0].toUpperCase() == "\"Err".toUpperCase()) {

}
}
});
} else {
alert("一次只能重打一行");
}
});
});

1.2 后台ajax代码

public class CreatedeliverRePrint_Ajax_ashx : IHttpHandler
{
string apiBaseUri = EncryptLib.DecryptString(System.Configuration.ConfigurationManager.AppSettings["Api_BaseUri"].ToString(), "ME Co,. Ltd.");
string userName = EncryptLib.DecryptString(System.Configuration.ConfigurationManager.AppSettings["Api_UserName"].ToString(), "ME Co,. Ltd.");
string password = EncryptLib.DecryptString(System.Configuration.ConfigurationManager.AppSettings["Api_PassWord"].ToString(), "ME Co,. Ltd.");
string clientId = EncryptLib.DecryptString(System.Configuration.ConfigurationManager.AppSettings["Api_ClientId"].ToString(), "ME Co,. Ltd.");
public void ProcessRequest(HttpContext context)
{
context.Response.ContentType = "text/plain";
string strCartonId = context.Request["CartonId"].ToString().Trim();
if (context.Request.Params.Count > 0)
{
var tokenStr = IndexApi.Adapter.GetToken(apiBaseUri, userName, password);
var result = IndexApi.Adapter.CallApiGet(apiBaseUri, "/api/zpl/Get/" + strCartonId, tokenStr, clientId);
context.Response.Write(result);
context.Response.End();
}
}

public bool IsReusable
{
get
{
return false;
}
}
}

2 使用post方式打印

2.1 前端页面js代码

function Print(strZplCommand) {
//var zpl = document.getElementById("zpl").value;
var zpl = strZplCommand;
var ip_addr = document.getElementById("hidIpAddress").value;
//var output = document.getElementById("output");
//var url = "http://" + ip_addr + "/pstprnt";

var url = "http://" + ip_addr + "/pstprnt";
var request = new Request(url, {
method: 'POST',
mode: 'no-cors',
cache: 'no-cache',
body: zpl
});

fetch(request)
.then(function (response) {

// Important : in 'no-cors' mode you are allowed to make a request but not to get a response
// even if in 'Network' Tab in chrome you can actually see printer response
// response.ok will always be FALSE and response.text() null
/*if (!response.ok) {
throw Error(response.statusText);
}*/
return true;
})

.catch(function (error) {

// note : if printer address is wrong or printer not available, you will get a "Failed to fetch"
console.log(error);

});
}

2.2 后端ajax代码

string apiBaseUri = EncryptLib.DecryptString(System.Configuration.ConfigurationManager.AppSettings["Api_BaseUri"].ToString(), "ME Co,. Ltd.");
string userName = EncryptLib.DecryptString(System.Configuration.ConfigurationManager.AppSettings["Api_UserName"].ToString(), "ME Co,. Ltd.");
string password = EncryptLib.DecryptString(System.Configuration.ConfigurationManager.AppSettings["Api_PassWord"].ToString(), "ME Co,. Ltd.");
string clientId = EncryptLib.DecryptString(System.Configuration.ConfigurationManager.AppSettings["Api_ClientId"].ToString(), "ME Co,. Ltd.");

context.Response.ContentType = "text/plain";
if (context.Request.Params.Count > 0)
{
DataTable dt = new DataTable();
dt.Columns.AddRange(new DataColumn[]{
new DataColumn("DeliveryOrderId",typeof(string))
,new DataColumn("PoId",typeof(string))
,new DataColumn("SeqNo",typeof(string))
,new DataColumn("Qty",typeof(string))
,new DataColumn("TransportQty",typeof(string))
,new DataColumn("VendorBatchNo",typeof(string))
});
DataRow row = null;

for (int i = 0; i < context.Request.Form.AllKeys.Count() / 6; i++)
{
row = dt.NewRow();
//row["OrderSeq"] = "450002241530";
row["DeliveryOrderId"] = context.Request.Form["boxs["+i+"][DeliveryOrderId]"];
row["PoId"] = context.Request.Form["boxs[" + i + "][PoId]"];
row["SeqNo"] = context.Request.Form["boxs[" + i + "][SeqNo]"];
row["Qty"] = context.Request.Form["boxs[" + i + "][Qty]"];
row["TransportQty"] = context.Request.Form["boxs[" + i + "][TransportQty]"];
row["VendorBatchNo"] = context.Request.Form["boxs[" + i + "][VendorBatchNo]"];
dt.Rows.Add(row);
}
dt.AcceptChanges();

#if DEBUG
dt.Clear();
DataRow rowTest = null;
rowTest = dt.NewRow();
rowTest["DeliveryOrderId"] = "20180125085905";
rowTest["PoId"] = "4500022415";
rowTest["SeqNo"] = "10";
rowTest["Qty"] = "47";
rowTest["TransportQty"] = "300";
rowTest["VendorBatchNo"] = "";
dt.Rows.Add(rowTest);

rowTest = dt.NewRow();
rowTest["DeliveryOrderId"] = "20180125085905";
rowTest["PoId"] = "4500022415";
rowTest["SeqNo"] = "30";
rowTest["Qty"] = "50";
rowTest["TransportQty"] = "300";
rowTest["VendorBatchNo"] = "";
dt.Rows.Add(rowTest);

dt.AcceptChanges();

#endif

string jsonData = JsonConvert.SerializeObject(dt);
jsonData = "{\"boxs\":"+jsonData;
jsonData = jsonData + "}";

// 调用接口
var tokenStr = IndexApi.Adapter.GetToken(apiBaseUri, userName, password);
var result = IndexApi.Adapter.CallApiPost(jsonData,apiBaseUri,"/api/zpl/post",tokenStr,clientId);

context.Response.Write(result);
context.Response.End();
}
}

public bool IsReusable
{
get
{
return false;
}
}

3、后端打印关键代码 RawPrinterHelper.cs

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;

using System.Text;
using System.IO;
using System.Drawing;
using System.Drawing.Printing;
using System.Runtime.InteropServices;

namespace BarCodePrintApi
{
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;
}

/// <summary>
/// 发送文件到打印机
/// </summary>
/// <param name="szPrinterName">打印机名称</param>
/// <param name="szFileName">文件</param>
/// <returns></returns>
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;
}

/// <summary>
/// 发送字符串到打印机
/// </summary>
/// <param name="szPrinterName">打印机名称</param>
/// <param name="szString">指令</param>
/// <returns></returns>
public static bool SendStringToPrinter1(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.
//Encoding.GetEncoding("GB2312").GetBytes(szString);
pBytes = Marshal.StringToCoTaskMemAnsi(szString);

// Send the converted ANSI string to the printer.
SendBytesToPrinter(szPrinterName, pBytes, dwCount);
Marshal.FreeCoTaskMem(pBytes);
return true;
}
/// <summary>
/// 打印标签带有中文字符的ZPL指令
/// </summary>
/// <param name="printerName">打印机名称</param>
/// <param name="szString">指令</param>
/// <returns></returns>
public static string SendStringToPrinter(string printerName, string szString)
{
byte[] bytes = Encoding.GetEncoding("GB2312").GetBytes(szString); //转换格式
IntPtr ptr = Marshal.AllocHGlobal(bytes.Length + 2);
try
{
Marshal.Copy(bytes, 0, ptr, bytes.Length);
SendBytesToPrinter(printerName, ptr, bytes.Length);
return "success";
}
catch(Exception ex)
{
return ex.Message;
}
finally
{
Marshal.FreeCoTaskMem(ptr);
}
}

/// <summary>
/// 网络打印
/// </summary>
/// <param name="ip"></param>
/// <param name="port"></param>
/// <param name="szString"></param>
/// <returns></returns>
public static string SendStringToNetPrinter(string ip, int port, string szString)
{
try
{
//打开连接
System.Net.Sockets.TcpClient client = new System.Net.Sockets.TcpClient();
client.Connect(ip, port);

//写入zpl命令
System.IO.StreamWriter writer = new System.IO.StreamWriter(client.GetStream());
writer.Write(szString);
writer.Flush();

//关闭连接
writer.Close();
client.Close();

return "success";
}
catch (Exception ex)
{
return ex.Message;
}
}
}
}

斑马打印机客户端GET和POST,以及后端两种打印方式。的更多相关文章

  1. 0404-服务注册与发现-客户端负载均衡-两种自定义方式-Ribbon通过代码自定义配置、使用配置文件自定义Ribbon Client

    一.官方文档解读 官方地址:https://cloud.spring.io/spring-cloud-static/Edgware.SR3/single/spring-cloud.html#_cust ...

  2. bootstrap table分页(前后端两种方式实现)

    bootstrap table分页的两种方式: 前端分页:一次性从数据库查询所有的数据,在前端进行分页(数据量小的时候或者逻辑处理不复杂的话可以使用前端分页) 服务器分页:每次只查询当前页面加载所需要 ...

  3. 吉特仓库管理系统- 斑马打印机 ZPL语言的腐朽和神奇

    上一篇文章说到了.NET中的打印机,在PrintDocument类也暴露一些本质上上的问题,前面也提到过了,虽然使用PrintDcoument打印很方便.对应条码打印机比如斑马等切刀指令,不依赖打印机 ...

  4. WebService客户端几种实现方式

    1.jdk原生调用(需要获取服务接口文件) import java.net.URL; import javax.xml.namespace.QName; import javax.xml.ws.Ser ...

  5. 以太坊go-ethereum客户端(三)两种全节点启动模式

    这篇博客介绍一下go-ethereum全节点的两种启动模式:主网络快速启动和测试网络快速启动.这也是客户端所提供的两种启动方式,直接拿来使用即可.下面具体介绍一下使用方法. 主网络快速启动 其实,我们 ...

  6. python实现斑马打印机网络打印

    最近一个礼拜调研了下斑马打印机怎样实现网络打印. 缘起: 之前实现打印方式是直接使用USB接口连接PC,使用串口通讯提供一套打印服务,在系统界面配置相关参数,即可调用打印服务: 后来业务需求变化,现场 ...

  7. C&num;调用斑马打印机打印条码标签&lpar;支持COM、LPT、USB、TCP连接方式和ZPL、EPL、CPCL指令&rpar;【转】

    原文地址:http://blog.csdn.net/ldljlq/article/details/7338772 在批量打印商品标签时一般都要加上条码或图片,而这类应用大多是使用斑马打印机,所以我也遇 ...

  8. 基于C&num;在WPF中使用斑马打印机进行打印【转】

    原文链接:http://ju.outofmemory.cn/entry/132476 最近在项目中接手了一个比较有挑战性的模块——用斑马打印机将需要打印的内容打印出来.苦苦折腾了两天,总算有所收获,就 ...

  9. 斑马打印机网卡ZebraNet配置(有线)

    实图: 抽象图: 说明: 1.并口,用于连接斑马打印机一端 2.网络连接状态指示灯 3.打印状态指示灯 4.测试按钮,在连接打印机的情况下,按下此键,会打印出网卡信息. 5.网线接口 按下测试按钮之后 ...

随机推荐

  1. CRL快速开发框架系列教程十&lpar;导出对象结构&rpar;

    本系列目录 CRL快速开发框架系列教程一(Code First数据表不需再关心) CRL快速开发框架系列教程二(基于Lambda表达式查询) CRL快速开发框架系列教程三(更新数据) CRL快速开发框 ...

  2. github上readme&period;md 格式

    参考:https://github.com/guoyunsky/Markdown-Chinese-Demo/edit/master/README.md

  3. IOS单例模式要做到3点

    1,永远只分配一块内存来创建对象. +(instanst) static id instace = nil; static dispatch_once_t onceToken; dispatch_on ...

  4. SDUT 1269 走迷宫(BFS)

    点我看题目 题意:中文不详述. 思路 :上上上场比赛让一个BFS给虐了,上次比赛让一个三维的给废掉了.......所以急于从水题刷起......还因为数组开小了WA了5,6次 #include &lt ...

  5. Less 编译工具

    Less 编译工具 虽然你可以选择在浏览器端使用Less,直接在页面中嵌入一个 Less.js 文件,你也可以选择在服务器端使用Less,使用命令行将Less文件编译成最终的CSS文件. 然而,这两种 ...

  6. Nginx服务器中配置非80端口的端口转发方法详解

    这篇文章主要介绍了Nginx服务器中配置非80端口的端口转发方法详解,文中使用到了Nginx中的proxy_pass配置项,需要的朋友可以参考下 nginx可以很方便的配置成反向代理服务器: 1 2 ...

  7. 11&period;17 flask &lpar;1&rpar;

    2018-11-17 18:38:42 开始学习进行玩前面项目  开始进军flask flask是一个小型的web框架,,但是有很多第三方组件 最后组装组装就和django一样啦!!!!!!! pyt ...

  8. hiho一下 第168周

    题目1 : 扩展二进制数 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 我们都知道二进制数的每一位可以是0或1.有一天小Hi突发奇想:如果允许使用数字2会发生什么事情? ...

  9. &lbrack;No0000180&rsqb;改善C&num;程序的建议8:避免锁定不恰当的同步对象

    在C#中让线程同步的另一种编码方式就是使用线程锁.所谓线程锁,就是锁住一个资源,使得应用程序只能在此刻有一个线程访问该资源.可以用下面这句不是那么贴切的话来理解线程锁的作用:锁,就是让多线程变成单线程 ...

  10. P2517 &lbrack;HAOI2010&rsqb;订货

    思路 费用流水题 对每月拆点,入点向出点连cap=ui的边,s向入点连cost=di的边,i的入点向i+1的入点连cap=S的边即可 代码 #include <cstdio> #inclu ...