揭破打字与印刷机驱动的心腹面纱,打印开发中关于打字与印刷协议

打印机打字与印刷协议有三种,EPOS和ESC/POS.

本文基于GP5八体系,它能够包容ESC/POS指令集,对EPSON的打字与印刷机通用.

佳博打印机代理商天猫店https://shop107172033.taobao.com/index.htm?spm=2013.1.w5002-9520741823.2.Sqz8Pf

Atitit.收银河系统pos
以及打字与印刷功用的行业标准

对于EPOS的打字与印刷开发,能够利用微软的库POS.NET举行支付(连接打字与印刷机供给动用逻辑名字,
能够在打字与印刷机官网下载相关的驱动),对于ESC/POS,能够行使串口开发(连接打字与印刷机要求使用com口名字,对于除com口之外的打字与印刷机供给在官网上下载相关的驱动)

Android下的装备调节和测试,倘诺设备提供了驱动,依据厂家的驱动调节和测试即可;设备未提供驱动,只可以根据通用的不二等秘书籍开始展览调剂。那里运用的是调用USB接口来决定打字与印刷机输出。

在此店购买的打字与印刷机,提供免费的技术服务和windows、android、ios开发指引

 

一.第一得到USB管理器

上一篇大家获得了经过打字与印刷驱动生成的十陆进制文件TestDriver如图四.1,接下去本身就来过跟各位看官具体分析一下那些十六进制数的意义。要打听这个十陆进制数据,有3个东东是不能缺少的,正是ESC/POS指令集,网上有下载的。以往境内多方的热敏打字与印刷都以用的ESC/POS指令,我们前面装的驱动就是遵照ESC/POS指令做的,用的打也是比照ESC/POS指令做的。打字与印刷机要和总结机要能实行对话,就得利用共同的语言,那种语言正是ESC/POS。驱动正是将微机要抒发的始末,“翻译”成ESC/POS指令,再告诉打字与印刷机,打字与印刷机就起来打字与印刷了。

 

public UsbAdmin(Context context) { 
    mUsbManager = (UsbManager) context.getSystemService(Context.USB_SERVICE); 
    mPermissionIntent = PendingIntent.getBroadcast(context, 0, new Intent(ACTION_USB_PERMISSION), 0); 
    IntentFilter filter = new IntentFilter(ACTION_USB_PERMISSION); 
    context.registerReceiver(mUsbReceiver, filter); 
  }

美高梅开户网址 1

一. ESC指令种类 Escape指令类别不一致于ESC/POS指令 1

选取二个推迟意图来接收usb接入时的广播,当播放接收到时,表达有新的设施联网。

图4.1

2. 打字与印刷标准OPOS POSPrinter 与 CashDrawer 驱动壹

累加一个boardcast action

     大家得以见到,第二行的数码一B40,那一个是十6进制数,约等于0x1B,0x40,大家再打开ESC/POS指令集,查找到一B40那条指令如图四.2,美高梅开户网址 2 
                                                                       
        图4.2

3. 命令集2

复制代码 代码如下:

能够看来那条指令十陆进为制一B 40,10进制为 27
64,十陆进制的0x1B也正是10进制的贰七,十6进制的40也便是10进制的6四.那条指令作用为开头化打字与印刷机。指令上面描述了命令的切切实实解释,首要是铲除打字与印刷缓冲区数据和NV位图数据。

4. Java框架jpos3

private static final String ACTION_USB_PERMISSION =
“com.android.example.USB_PERMISSION”;

      再看下一条,1B肆A30,打开ESC/POS指令集,如图四.三

5. jpos.JposException: Service does not exist in loaded JCL registry3

private final BroadcastReceiver mUsbReceiver = new BroadcastReceiver() { 
    public void onReceive(Context context, Intent intent) { 
      String action = intent.getAction(); 
      if (ACTION_USB_PERMISSION.equals(action)) { 
        synchronized (this) { 
          UsbDevice device = (UsbDevice) intent.getParcelableExtra(UsbManager.EXTRA_DEVICE); 
          if (intent.getBooleanExtra(UsbManager.EXTRA_PERMISSION_GRANTED, false)) { 
            if (device != null) { 
              setDevice(device); 
            } else { 
              Closeusb(); 
             // mDevice = device; 
            } 
          } else { 
            Log.d(TAG, "permission denied for device " + device); 
          } 

        } 

      } 
    } 
  }; 

美高梅开户网址 3

5.1.1. (JCL) Jar Class Loader 3

取到usb设备的引用,android系统会询问你是或不是允许设备访问,暗中同意为false;当允许了访问之后,会判定USB的引用是或不是为null,借使不为空则会调用setDevice来创造八个Connection,否则会倒闭此番连接。

如图4.3

 

在setDevice中,大家能够收获装备的功力集(UsbInterface),也足以拿走通讯通道(UsbEndpoint),同时也开创了host与device的连接用来传输数据。

能够见到那条命令的机能是打字与印刷并走纸,那条命令还含有贰个参数n,以往下令为一B四A30,n就为十陆进制0x30约等于10进制4八,所以依照指令描述,走纸的偏离为
n x
纵向或横向移动单位,那个纵向或横向移动单位,是能够安装的,1般打字与印刷机暗中认可为3个点也便是0.125mm,所以走纸距离为48x 0.125mm = 6mm。

 

private void setDevice(UsbDevice device) { 
    if (device != null) { 
      UsbInterface intf = null; 
      UsbEndpoint ep = null; 

      int InterfaceCount = device.getInterfaceCount(); 
      int j; 

      mDevice = device; 
      for (j = 0; j < InterfaceCount; j++) { 
        int i; 

        intf = device.getInterface(j); 
        Log.i(TAG, "接口是:" + j + "类是:" + intf.getInterfaceClass()); 
        if (intf.getInterfaceClass() == 7) { 
          int UsbEndpointCount = intf.getEndpointCount(); 
          for (i = 0; i < UsbEndpointCount; i++) { 
            ep = intf.getEndpoint(i); 
            Log.i(TAG, "端点是:" + i + "方向是:" + ep.getDirection() + "类型是:" + ep.getType()); 
            if (ep.getDirection() == 0 && ep.getType() == UsbConstants.USB_ENDPOINT_XFER_BULK) { 
              Log.i(TAG, "接口是:" + j + "端点是:" + i); 
              break; 
            } 
          } 
          if (i != UsbEndpointCount) { 
            break; 
          } 
        } 
      } 
      if (j == InterfaceCount) { 
        Log.i(TAG, "没有打印机接口"); 
        return; 
      } 

      mEndpointIntr = ep; 

        UsbDeviceConnection connection = mUsbManager.openDevice(device); 

        if (connection != null && connection.claimInterface(intf, true)) { 
          Log.i(TAG, "打开成功! "); 
          mConnection = connection; 

        } else { 
          Log.i(TAG, "打开失败! "); 
          mConnection = null; 
        } 
      } 

  } 

     接下来一条为一D7陆三千3001八….那条指令相比复杂,查看指令如图四.四

壹. ESC指令行列 Escape指令系列差异于ESC/POS指令 

揭破打字与印刷机驱动的心腹面纱,打印开发中关于打字与印刷协议。Escape指令种类: 是UPOS协会规范提议的国际 零售设备调用标准! 

 

ESC/POS指令集: 是EPSON在原来的ESC/P指令 ESC/P 系统基础上升心花怒放起的,系统设备使用指令集合与 行业标准!

 

该打印控制命令(WPSON StandardCode for Printer)是EPSON公司团结制定的针式打字与印刷机的基准指令集,现在已变为针式打字与印刷机控制语言事实上的工业标准。ESC/POS打字与印刷命令集是ESC打印控制命令的简化版本,未来多数票据打印都施用ESC/POS指令集。其鲜明特色是:个中十分大学一年级些限令都以以ESC控制符发轫的1串代码。

 

笔者:: 老哇的爪子 Attilax 艾龙,  EMAIL:14665一9八四玖@qq.com

转发请注脚来源: 

 

 

贰.在有关的类中新建三个UsbAdmin,调用openUsb,那里首先是走了地点的setDevice()方法,获取到了设备的引用,当连接通道建立刻列出具有USB设备,当设备的引用不设有时1样列出富有的USB设备,并且都呼吁获取USB权限。

美高梅开户网址 4

二.  打印标准OPOS POSPrinter 与 CashDrawer 驱动

OPOS (OLE for Point Of Sale) 驱动, 援救 POSPrinter 及 CashDrawer 设备连串.
OPOS 是建基于 Microsoft ActiveX 架构的驱动系统. OPOS 的含义在于令用者能够在指标导向环境上边使用票据打字与印刷机和钱箱的装有作用.

 

public void openUsb() { 
    if (mDevice != null) { 
      setDevice(mDevice); 
      if (mConnection == null) { 
        HashMap<String, UsbDevice> deviceList = mUsbManager.getDeviceList(); 
        Iterator<UsbDevice> deviceIterator = deviceList.values().iterator(); 

        while (deviceIterator.hasNext()) { 
          UsbDevice device = deviceIterator.next(); 
          mUsbManager.requestPermission(device, mPermissionIntent); 
        } 
      } 
    } else { 
      HashMap<String, UsbDevice> deviceList = mUsbManager.getDeviceList(); 
      Iterator<UsbDevice> deviceIterator = deviceList.values().iterator(); 

      while (deviceIterator.hasNext()) { 
        UsbDevice device = deviceIterator.next(); 
        mUsbManager.requestPermission(device, mPermissionIntent); 
      } 
    } 
  } 

美高梅开户网址 5

3. 命令集

英文形式下的命令 
代码 功能 
CR 回车 
ESC ! 设置打字与印刷情势 
ESC % 选择或吊销用户自定义字符集 
ESC & 定义用户自定义字符集 
ESC * 设置位映射格局 
ESC @ 起初化打字与印刷机 
ESC ~ LED ON/OFF 
ESC < 重回行首 
ESC 二 采用行间距为百分之十六英寸 
ESC 三 设置行进为最小间距 
ESC BEL 蜂鸣器ON/OFF 
ESC C 设置单页长度 
ESC c0 选取打字与印刷页 
ESC c1 选取行间距 
ESC c三 选拔纸甘休非确定性信号输出 
ESC c四 选用打字与印刷纸及检查实验器(终止打字与印刷) 
美高梅开户网址,ESC c5 禁止/使能面板开关 
ESC c6 禁止/使能ON-LINE开关 
ESC d 打印及N行进纸 
ESC D 设置TAB位置 
ESC e 打印病退回N行 
ESC f 设单页等待时间 
ESC F 采取或吊销单页退纸区 
ESC i 全切割 
ESC J 以细小间距进行打字与印刷和进纸 
ESC K 以细小间距实行打字与印刷和退纸 
ESC l 选取或撤废倒过来的字符 
ESC m 局地切割 
ESC o 印章 
ESC p 发生钦赐脉冲 
ESC q 释放纸 
ESC r 选取打字与印刷颜色 
ESC 奥德赛 选取国际字符子集 
ESC SP 设置左边界 
ESC t 接纳字符码表 
ESC U 选拔或吊销单向打字与印刷 
ESC V 发送打字与印刷机状态 
ESC z 设置或打消两页并行打印 
FF 打字与印刷送出单页 
HT 水平TAB 
LF 换行 
RS 流水TAB

3.当下边两部都走完了后来,我们就足以发送指令来控制已经确立连接的打字与印刷机了,那里大家使用的是规范的ESC/POS指令集,为硬件私下认可,贴出代码,那里的命令集应用的是10进制表示方式,也能够替换来十陆进制。

   
能够见到那条指令是用来打字与印刷光栅位图的,参数相比较多,大家逐条来对号入座,对照大家的多少一D7陆贰仟三千1八…后,获得m=00,xL=0x30,xH=0x00,yL=0x1八,yH=0x00,k=115二,这几个命令正是驱动翻译的显要部分,是将图文根据这几个命令规定的法子,转换为点阵,传给打字与印刷机,打字与印刷机接收到那一个点阵后,将图纸和文字打字与印刷出来。

4. Java框架jpos

 

public class printerCmdUtils { 

  /** 
   * 这些数据源自爱普生指令集,为POS机硬件默认 
   */ 

  public static final byte ESC = 27;//换码 
  public static final byte FS = 28;//文本分隔符 
  public static final byte GS = 29;//组分隔符 
  public static final byte DLE = 16;//数据连接换码 
  public static final byte EOT = 4;//传输结束 
  public static final byte ENQ = 5;//询问字符 
  public static final byte SP = 32;//空格 
  public static final byte HT = 9;//横向列表 
  public static final byte LF = 10;//打印并换行(水平定位) 
  public static final byte CR = 13;//归位键 
  public static final byte FF = 12;//走纸控制(打印并回到标准模式(在页模式下) ) 
  public static final byte CAN = 24;//作废(页模式下取消打印数据 ) 



//------------------------打印机初始化----------------------------- 


  /** 
   * 打印机初始化 
   * @return 
   */ 
  public static byte[] init_printer() 
  { 
    byte[] result = new byte[2]; 
    result[0] = ESC; 
    result[1] = 64; 
    return result; 
  } 


//------------------------换行----------------------------- 


  /** 
   * 换行 
   * @param lineNum要换几行 
   * @return 
   */ 
  public static byte[] nextLine(int lineNum) 
  { 
      byte[] result = new byte[lineNum]; 
      for(int i=0;i<lineNum;i++) 
      { 
        result[i] = LF; 
      } 

      return result; 
  } 


//------------------------下划线-----------------------------   


  /** 
   * 绘制下划线(1点宽) 
   * @return 
   */ 
  public static byte[] underlineWithOneDotWidthOn() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 45; 
    result[2] = 1; 
    return result; 
  } 


  /** 
   * 绘制下划线(2点宽) 
   * @return 
   */ 
  public static byte[] underlineWithTwoDotWidthOn() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 45; 
    result[2] = 2; 
    return result; 
  } 
  /** 
   * 取消绘制下划线 
   * @return 
   */ 
  public static byte[] underlineOff() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 45; 
    result[2] = 0; 
    return result; 
  } 


//------------------------加粗----------------------------- 


  /** 
   * 选择加粗模式 
   * @return 
   */ 
  public static byte[] boldOn() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 69; 
    result[2] = 0xF; 
    return result; 
  } 


  /** 
   * 取消加粗模式 
   * @return 
   */ 
  public static byte[] boldOff() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 69; 
    result[2] = 0; 
    return result; 
  } 


//------------------------对齐----------------------------- 


  /** 
   * 左对齐 
   * @return 
   */ 
  public static byte[] alignLeft() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 97; 
    result[2] = 0; 
    return result; 
  } 


  /** 
   * 居中对齐 
   * @return 
   */ 
  public static byte[] alignCenter() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 97; 
    result[2] = 1; 
    return result; 
  } 


  /** 
   * 右对齐 
   * @return 
   */ 
  public static byte[] alignRight() 
  { 
      byte[] result = new byte[3]; 
    result[0] = ESC; 
    result[1] = 97; 
    result[2] = 2; 
    return result; 
  } 


  /** 
   * 水平方向向右移动col列 
   * @param col 
   * @return 
   */ 
  public static byte[] set_HT_position( byte col ) 
  { 
    byte[] result = new byte[4]; 
    result[0] = ESC; 
    result[1] = 68; 
    result[2] = col; 
    result[3] = 0; 
    return result; 
  } 
//------------------------字体变大----------------------------- 


  /** 
   * 字体变大为标准的n倍 
   * @param num 
   * @return 
   */ 
  public static byte[] fontSizeSetBig(int num) 
  { 
      byte realSize = 0; 
      switch (num) 
      { 
      case 1: 
        realSize = 0;break; 
      case 2: 
        realSize = 17;break; 
      case 3: 
        realSize = 34;break; 
      case 4: 
        realSize = 51;break; 
      case 5: 
        realSize = 68;break; 
      case 6: 
        realSize = 85;break; 
      case 7: 
        realSize = 102;break; 
      case 8: 
        realSize = 119;break; 
      } 
      byte[] result = new byte[3]; 
      result[0] = 29; 
      result[1] = 33; 
      result[2] = realSize; 
      return result; 
  } 


//------------------------字体变小----------------------------- 


  /** 
   * 字体取消倍宽倍高 
   * @param num 
   * @return 
   */ 
  public static byte[] fontSizeSetSmall(int num) 
  { 
      byte[] result = new byte[3]; 
      result[0] = ESC; 
      result[1] = 33; 

    return result; 
  } 


//------------------------切纸-----------------------------   


  /** 
   * 进纸并全部切割 
   * @return 
   */ 
  public static byte[] feedPaperCutAll() 
  { 
      byte[] result = new byte[4]; 
     result[0] = GS; 
     result[1] = 86; 
     result[2] = 65; 
     result[3] = 0; 
     return result; 
  } 


  /** 
   * 进纸并切割(左边留一点不切) 
   * @return 
   */ 
  public static byte[] feedPaperCutPartial() 
  { 
      byte[] result = new byte[4]; 
     result[0] = GS; 
     result[1] = 86; 
     result[2] = 66; 
     result[3] = 0; 
     return result; 
  } 

//------------------------切纸----------------------------- 
  public static byte[] byteMerger(byte[] byte_1, byte[] byte_2){  
    byte[] byte_3 = new byte[byte_1.length+byte_2.length];  
    System.arraycopy(byte_1, 0, byte_3, 0, byte_1.length);  
    System.arraycopy(byte_2, 0, byte_3, byte_1.length, byte_2.length);  
    return byte_3;  
  }  


  public static byte[] byteMerger(byte[][] byteList){  

      int length = 0; 
    for(int i=0;i<byteList.length;i++) 
    { 
        length += byteList[i].length; 
    } 
    byte[] result = new byte[length]; 

    int index = 0; 
    for(int i=0;i<byteList.length;i++) 
    { 
        byte[] nowByte = byteList[i]; 
        for(int k=0;k<byteList[i].length;k++) 
        { 
          result[index] = nowByte[k]; 
          index++; 
        } 
    } 
    return result;  
  }  



} 

   
从地点的辨析能够见到驱动其实没干什么神奇的事,只是即将打字与印刷的剧情,转换为多少点阵,再发放打字与印刷机,再打字与印刷出来,那几个数量点阵的格式都以根据ESC/POS指令,所以说打字与印刷机开发中最关键的正是掌握ESC/POS指令集,后边笔者会日益介绍ESC/POS指令。

5. jpos.JposException: Service does not exist in loaded JCL registry

四.在上述都做到之后,就能够把你须要的字符串转换到byte数组并调用sendCommand方法来进行打字与印刷了

5.1.1. (JCL) Jar Class Loader 

 

使用javapos必要掌握打字与印刷机与钱箱名称。。。要设置个布局文件。。麻烦。

 

 

 

参考

 

jpos_examples_图文_百度文库.htm

ESC POS 命令 用法_百度知道.htm

esc pos打字与印刷指令 (小寿转发)-xiaoshou330-ChinaUnix博客.htm

@SuppressLint("NewApi") 
  public boolean sendCommand(byte[] Content) { 
    boolean Result; 
    synchronized (this) { 
      int len = -1; 
      if (mConnection != null) { 
        len = mConnection.bulkTransfer(mEndpointIntr, Content, Content.length, 10000); 
      } 

      if (len < 0) { 
        Result = false; 
        Log.i(TAG, "发送失败! " + len); 
      } else { 
        Result = true; 
        Log.i(TAG, "发送" + len + "字节数据"); 
      } 
    } 
    return Result; 

复制代码 代码如下:

len = mConnection.bulkTransfer(mEndpointIntr, Content, Content.length,
10000); 

这一步仅仅加了合伙锁,并未有开启多少个新的线程去处理,在本机上尚无难点,但下边的USB通讯机制的稿子有涉嫌要放权异步线程,那里供给注意。

以上便是本文的全体内容,希望对大家的就学抱有支持,也意在大家多多扶助脚本之家。

你只怕感兴趣的稿子:

  • Android 蓝牙( Bluetooth® )连接 ESC/POS
    热敏打字与印刷机打字与印刷实例(ESC/POS指令篇)
  • Android 蓝牙( Bluetooth® )连接 ESC/POS
    热敏打字与印刷机打印实例(蓝牙( Bluetooth® )连接篇)
  • Android进阶——安卓调用ESC/POS打印机打字与印刷实例

发表评论

电子邮件地址不会被公开。 必填项已用*标注

网站地图xml地图