最近一个礼拜调研了下斑马打印机怎样实现网络打印。

  缘起:

    之前实现打印方式是直接使用USB接口连接PC,使用串口通讯提供一套打印服务,在系统界面配置相关参数,即可调用打印服务;

    后来业务需求变化,现场实施并没有PC提供给打印机使用USB连接方式,因此,就开始做了这件事。

  调研后方案:

  硬件:一台Zebra ZT210斑马打印机、一个USR W610模块、一根网线

  方案一:

    后端发送ZPL指令到打印机,封装统一调用

  方案二(舍弃):

    使用本地斑马打印机驱动调用打印机,普通打印Ctrl + P调起的界面(前端实现,此种方式有几个难点:1.格式调整;2.二维码或者条码的生成,但这对于前端也不是什么难点,有实现的,相比下后端实现简单)

  方案一实现过程:

  

 

 

  打印流程:

 

   有人的USR W610模块实物(天线可以忽略,本次使用的网线,网口在电源线旁边):

  USR W610:充当一个TCP Server,上电后,插上网线,输入模块后面的ip(账号密码:admin/admin)就可以访问它的管理界面。若之前设置过ip,找根笔戳一下重置按钮。设置ip和端口,后端使用socket进行连接时会使用。

  USR W610后台界面:

 

·     后端核心代码:

    def post(self, request):
        """新增对象

        Args:
            request (rest_framework.request.Request): HTTP request

        Returns:
            response(rest_framework.response.Response): HTTP response.
            error_response(rest_framework.response.error_response): error_response

        """
        request_data = request.data
        template_name = request_data.get('template_name', '')
        if not template_name:
            return error_response(reason='模板名称不能为空',
                                  info='template_name is required',
                                  state=status.HTTP_400_BAD_REQUEST)

        function_name = request_data.get('function_name', '')
        if not function_name:
            return error_response(reason='功能模块名称不能为空',
                                  info='function_name is required',
                                  state=status.HTTP_400_BAD_REQUEST)

        is_function_name_exist = PrinterConfig.objects.filter(function_name=function_name).exists()
        if not is_function_name_exist:
            return error_response(reason='功能模块名称不存在',
                                  info='function_name is not exist',
                                  state=status.HTTP_400_BAD_REQUEST)

        is_template_name_exist = PrinterConfig.objects.filter(template_name=template_name).exists()
        current_dir = os.path.join(settings.BASE_DIR, 'print_template')
        is_template_name_in = template_name in os.listdir(current_dir)
        if not any([is_template_name_in, is_template_name_exist]):
            return error_response(reason='模板名称不存在',
                                  info='template_name is not exist',
                                  state=status.HTTP_400_BAD_REQUEST)

        file_path = os.path.join(current_dir, template_name)
        qr_code_file = open(file_path, 'r', encoding='utf-8')
        template_data = qr_code_file.read()
        template = Template(template_data)
        zpl = template.render(Context(request_data))

        # 携带ZPL指令向打印机发送http请求
        ip_port = PrinterConfig.objects.filter(template_name=template_name).values_list('printer_ip', 'printer_port')
        ip, port = ip_port[0][0], ip_port[0][1]
        if not all([ip, port]):
            return error_response(reason='打印配置的IP或端口未配置',
                                  info='printer ip and port must be configured',
                                  state=status.HTTP_400_BAD_REQUEST)

        # print_server = 'http://' + str(ip) + ':' + str(port)
        client = None
        try:
            # requests.post(print_server, data=zpl.encode('utf-8'))
            client = socket.socket(socket.AF_INET, type=socket.SOCK_STREAM)
            client.connect((ip, port))
            client.send(zpl.encode('utf-8'))
            # 打印机未连接!
        except Exception as other_except:  # pylint: disable=broad-except
            except_info = other_except.args[0]
            # except_info = other_except.args[0].args[0]
            # if except_info == 'Connection aborted.':
            if isinstance(except_info, tuple) and except_info == 'Connection aborted.':
                return Response({'result': '打印成功,请确认'})
            else:
                logger.error(other_except)
                return error_response(reason='打印失败,请检查打印配置是否正确',
                                      info=str(other_except),
                                      state=status.HTTP_400_BAD_REQUEST)
        finally:
            client.close()

        return Response({'result': 'ok'})

  特殊说明:

  1、正如代码所表现的,打印机不会给模块响应,模块也就不会给后端响应,打印成功会抛Connection aborted.异常,实际已经打印出来。但这里为什么要请确认,是因为在TCP连接正常情况下,即使把耗材取出,比如把标签纸拿出去,也会打印成功,等换上纸后,打印任务队列对接着打印,所以这个就需要现场人员确认了。

  2、模板文件里面的ZPL指令编写,可以参考http://note.youdao.com/noteshare?id=05f00edb5f88cfe16543337f8c7f17aa&sub=77F69DD3BA7E4961A3435E9DFA7D15E5 也可以使用Zebra Designer工具进行设计生成.prn文件,文本打开即可看见ZPL指令。因为自动生成的.prn文件中ZPL指令是经过GFA加密过的,不便于使用模板语法替换,生成的内容也相比自己写的多很多。工具界面如下:

 

 

 

 .prn生成的zpl指令示例:

 

 自己参考ZPL指令手册写的指令示例:

^XA
^CI28
^CW1,E:SIMSUN.TTF
^MD20
~SD20
^FO142,105
^BQN,2,10
^FD  {{qr_code_print}}
^FS
^XZ

 

   其他说明:zpl指令中有两个值得注意的:SD   设置暗度:若打印字迹比较淡时设置            PR打印速率:若打印字迹比较稀时设置

参考资料:

  https://www.cnblogs.com/chengeng/p/7676046.html

  https://max.book118.com/html/2018/1006/8002046103001125.shtm

  https://www.usr.cn/Down/USR-W610_instructions.pdf

   

posted on 2021-02-26 11:23  落叶虽美只活一世  阅读(3341)  评论(0编辑  收藏  举报
Live2D