python连接打印机实现打印文档、图片、pdf文件等功能


Posted in Python onFebruary 07, 2020

引言

python连接打印机进行打印,可能根据需求的不同,使用不同的函数模块。

  1. 如果你只是简单的想打印文档,比如office文档,你可以使用ShellExecute方法,对于微软office的文档、pdf、txt等有用,你可以尝试下;
  2. 如果你输入某些数据,文字信息,就想直接把它发送给打印机打印,那么可以尝试使用win32print;
  3. 如果你有一张图片,那么你可以结合python的Python Imaging Library(PIL)win32ui模块进行打印;

普通打印

ShellExecute

  • 首先确保你电脑中的应用可以打开你要打印的文件;
  • 是一些标准的文件类型
  • 不用管哪些打印机,也就是说和连接的打印机型号无关;
  • 你无控制设置打印属性的权限;
import tempfile
import win32api
import win32print

filename = tempfile.mktemp (".txt")
open (filename, "w").write ("This is a test")
win32api.ShellExecute (
 0,
 "print",
 filename,
 #
 # If this is None, the default printer will
 # be used anyway.
 #
 '/d:"%s"' % win32print.GetDefaultPrinter (),
 ".",
 0
)

另一个版本

import tempfile
import win32api
import win32print

filename = tempfile.mktemp (".txt")
open (filename, "w").write ("This is a test")
win32api.ShellExecute (
 0,
 "printto",
 filename,
 '"%s"' % win32print.GetDefaultPrinter (),
 ".",
 0
)

直接打印数据

win32print

  • 直接将数据扔给打印机;
  • 快速而且容易;
  • 而且可以定义选择哪个打印机打印;
  • 但是要打印的数据必须是可打印的,例如字符串等;
import os, sys
import win32print
printer_name = win32print.GetDefaultPrinter ()
#
# raw_data could equally be raw PCL/PS read from
# some print-to-file operation
#
if sys.version_info >= (3,):
 raw_data = bytes ("This is a test", "utf-8")
else:
 raw_data = "This is a test"

hPrinter = win32print.OpenPrinter (printer_name)
try:
 hJob = win32print.StartDocPrinter (hPrinter, 1, ("test of raw data", None, "RAW"))
 try:
  win32print.StartPagePrinter (hPrinter)
  win32print.WritePrinter (hPrinter, raw_data)
  win32print.EndPagePrinter (hPrinter)
 finally:
  win32print.EndDocPrinter (hPrinter)
finally:
 win32print.ClosePrinter (hPrinter)

打印图片

PIL win32ui

不使用额外的工具,在windows电脑上打印一张图片是相当的困难,至少需要3种不同的且相关的设备环境才可以。
还好,device-independent bitmap(DIB)和PIL可以帮助我们快速打印。下面的代码可以将图片发送至打印机打印尽可能大的尺寸且不失比例。

  • 还可以选择使用哪个打印机
  • 选择加载的图片的格式等
  • 但是如果你电脑不是windows,那可能不是最好的方法;
import win32print
import win32ui
from PIL import Image, ImageWin

#
# Constants for GetDeviceCaps
#
#
# HORZRES / VERTRES = printable area
#
HORZRES = 8
VERTRES = 10
#
# LOGPIXELS = dots per inch
#
LOGPIXELSX = 88
LOGPIXELSY = 90
#
# PHYSICALWIDTH/HEIGHT = total area
#
PHYSICALWIDTH = 110
PHYSICALHEIGHT = 111
#
# PHYSICALOFFSETX/Y = left / top margin
#
PHYSICALOFFSETX = 112
PHYSICALOFFSETY = 113

printer_name = win32print.GetDefaultPrinter ()
file_name = "test.jpg"

#
# You can only write a Device-independent bitmap
# directly to a Windows device context; therefore
# we need (for ease) to use the Python Imaging
# Library to manipulate the image.
#
# Create a device context from a named printer
# and assess the printable size of the paper.
#
hDC = win32ui.CreateDC ()
hDC.CreatePrinterDC (printer_name)
printable_area = hDC.GetDeviceCaps (HORZRES), hDC.GetDeviceCaps (VERTRES)
printer_size = hDC.GetDeviceCaps (PHYSICALWIDTH), hDC.GetDeviceCaps (PHYSICALHEIGHT)
printer_margins = hDC.GetDeviceCaps (PHYSICALOFFSETX), hDC.GetDeviceCaps (PHYSICALOFFSETY)

#
# Open the image, rotate it if it's wider than
# it is high, and work out how much to multiply
# each pixel by to get it as big as possible on
# the page without distorting.
#
bmp = Image.open (file_name)
if bmp.size[0] > bmp.size[1]:
 bmp = bmp.rotate (90)

ratios = [1.0 * printable_area[0] / bmp.size[0], 1.0 * printable_area[1] / bmp.size[1]]
scale = min (ratios)

#
# Start the print job, and draw the bitmap to
# the printer device at the scaled size.
#
hDC.StartDoc (file_name)
hDC.StartPage ()

dib = ImageWin.Dib (bmp)
scaled_width, scaled_height = [int (scale * i) for i in bmp.size]
x1 = int ((printer_size[0] - scaled_width) / 2)
y1 = int ((printer_size[1] - scaled_height) / 2)
x2 = x1 + scaled_width
y2 = y1 + scaled_height
dib.draw (hDC.GetHandleOutput (), (x1, y1, x2, y2))

hDC.EndPage ()
hDC.EndDoc ()
hDC.DeleteDC ()

实践

从前台传来要打印的字符,后端生成二维码,并作出相应处理后,连接打印机打印图片。

# 打印二维码
def print_barcode(request):
  import pyqrcode
  import random,string
  from PIL import Image,ImageDraw,ImageFont
  import numpy as np
  if request.is_ajax() and request.method == 'POST':
    result = {}
    bar_string = 'NaN'
    type = request.POST['type']

    if type == 'box':
      # 生成箱子码
      # 格式:P190823-K91 [P][日期][-][A-Z][0-9][0-9]
      bar_string = 'P'+datetime.date.today().strftime('%y%m%d')+'-'+str(random.choice('ABCDEFGHIGKLMNOPQRSTUVWXYZ'))\
             + str(random.choice(range(10)))+ str(random.choice(range(10)))
    elif type == 'kuwei':
      # 生成库位码
      bar_string = request.POST['string']
    else:
      pass

    try:
      big_code = pyqrcode.create(bar_string, error='L', version=2 , mode='binary')
      big_code.png('./code.png', scale=8)
      img_code = Image.open('code.png')

      size = img_code.size
      img_final = Image.new('RGB', (size[0], size[1]+35), color=(255, 255, 255))
      img_final.paste(img_code, (0, 0, size[0], size[1]))

      draw = ImageDraw.Draw(img_final)
      font = ImageFont.truetype('AdobeGothicStd-Bold.otf', size=35)
      width, height = draw.textsize(bar_string,font=font)
      draw.text(((size[0]-width)/2, size[1]-15), bar_string , fill=(0, 0, 0), font=font)
      img_final.save('./code.png')

      # 然后连接打印机将其打印出来即可
      is_ok =[]
      if type == 'box':
        for i in range(4):
          temp = print_img('./code.png')
          is_ok.append(temp)
      else:
        temp = print_img('./code.png')
        is_ok.append(temp)
      # is_ok = True
      result['done'] = 'ok' if np.all(is_ok) else '连接打印机失败'
    except Exception as e:
      result['done'] = e

    return JsonResponse(result)
 
 def print_img(img):
  import win32print
  import win32ui
  from PIL import Image, ImageWin
  # 参考 http://timgolden.me.uk/python/win32_how_do_i/print.html#win32print
  try:
    printer_name = win32print.GetDefaultPrinter()
    hDC = win32ui.CreateDC()
    hDC.CreatePrinterDC(printer_name)

    #printable_area = (300, 270) # 打印纸尺寸
    #printer_size = (300, 270)

    # 打开图片并缩放
    bmp = Image.open(img)
    if bmp.size[0] < bmp.size[1]:
      bmp = bmp.rotate(90)

    # ratios = [1.0 * printable_area[0] / bmp.size[1], 1.0 * printable_area[1] / bmp.size[0]]
    # scale = min(ratios)
    scale = 1

    hDC.StartDoc(img)
    hDC.StartPage()

    dib = ImageWin.Dib(bmp)
    scaled_width, scaled_height = [int(scale * i) for i in bmp.size]

    x1 = 20 # 控制位置
    y1 = -30
    x2 = x1 + scaled_width
    y2 = y1 + scaled_height
    dib.draw(hDC.GetHandleOutput(), (x1, y1, x2, y2))

    hDC.EndPage()
    hDC.EndDoc()
    hDC.DeleteDC()

    return True
  except:
    return False

打印效果:

python连接打印机实现打印文档、图片、pdf文件等功能

以上内容为二赛君整理发布,转载请注明出处,谢谢。

参考

http://timgolden.me.uk/python/win32_how_do_i/print.htm

Python 相关文章推荐
python 示例分享---逻辑推理编程解决八皇后
Jul 20 Python
把项目从Python2.x移植到Python3.x的经验总结
Apr 20 Python
详细解读Python的web.py框架下的application.py模块
May 02 Python
浅谈python正则的常用方法 覆盖范围70%以上
Mar 14 Python
使用python实现mqtt的发布和订阅
May 05 Python
Pandas之ReIndex重新索引的实现
Jun 25 Python
python同义词替换的实现(jieba分词)
Jan 21 Python
Python计算公交发车时间的完整代码
Feb 12 Python
python通过文本在一个图中画多条线的实例
Feb 21 Python
Python *args和**kwargs用法实例解析
Mar 02 Python
浅谈TensorFlow中读取图像数据的三种方式
Jun 30 Python
用Python实现童年贪吃蛇小游戏功能的实例代码
Dec 07 Python
解决Keras 与 Tensorflow 版本之间的兼容性问题
Feb 07 #Python
tensorflow2.0与tensorflow1.0的性能区别介绍
Feb 07 #Python
python第三方库学习笔记
Feb 07 #Python
Python字典添加,删除,查询等相关操作方法详解
Feb 07 #Python
tensorflow之自定义神经网络层实例
Feb 07 #Python
在tensorflow中设置使用某一块GPU、多GPU、CPU的操作
Feb 07 #Python
谈一谈数组拼接tf.concat()和np.concatenate()的区别
Feb 07 #Python
You might like
ThinkPHP视图查询详解
2014/06/30 PHP
PHP判断浏览器、判断语言代码分享
2015/03/05 PHP
PHP实现清除MySQL死连接的方法
2016/07/23 PHP
ext监听事件方法[初级篇]
2008/04/27 Javascript
WEB高性能开发之疯狂的HTML压缩
2010/06/19 Javascript
js实现iframe自动自适应高度的方法
2015/02/17 Javascript
去除html代码里面的script正则方法
2016/05/19 Javascript
jQuery处理XML文件的几种方法
2016/06/14 Javascript
深入理解React中es6创建组件this的方法
2016/08/29 Javascript
js实现各种复制到剪贴板的方法(分享)
2016/10/27 Javascript
浅谈Vuejs中nextTick()异步更新队列源码解析
2017/12/31 Javascript
在element-ui的el-tree组件中用render函数生成el-button的实例代码
2018/11/05 Javascript
JavaScript时间与时间戳的转换操作实例分析
2018/12/07 Javascript
vue-cli的build的文件夹下没有dev-server.js文件配置mock数据的方法
2019/04/17 Javascript
Vue.js + Nuxt.js 项目中使用 Vee-validate 表单校验
2019/04/22 Javascript
Easyui 关闭jquery-easui tab标签页前触发事件的解决方法
2019/04/28 jQuery
解决vue单页面修改样式无法覆盖问题
2019/08/05 Javascript
vue-resourc发起异步请求的方法
2020/02/11 Javascript
Node.js API详解之 readline模块用法详解
2020/05/22 Javascript
[02:25]专访DOTA2负责人Erik 国际邀请赛暂不会离开西雅
2014/07/21 DOTA
python中的字典详细介绍
2014/09/18 Python
Python中使用glob和rmtree删除目录子目录及所有文件的例子
2014/11/21 Python
python中os模块详解
2016/10/14 Python
Python程序退出方式小结
2017/12/09 Python
Python实现读取字符串按列分配后按行输出示例
2018/04/17 Python
Pycharm2020.1安装中文语言插件的详细教程(不需要汉化)
2020/08/07 Python
HTML5拖拽API经典实例详解
2018/04/20 HTML / CSS
PHP经典面试题
2016/09/03 面试题
七年级生物教学反思
2014/01/30 职场文书
2014年创卫实施方案
2014/02/18 职场文书
美术指导助理求职信
2014/04/20 职场文书
企业金融服务方案
2014/06/03 职场文书
2015年度内部审计工作总结
2015/05/20 职场文书
Python 中的单分派泛函数你真的了解吗
2021/06/22 Python
关于nginx 实现jira反向代理的问题
2021/09/25 Servers
CSS中妙用 drop-shadow 实现线条光影效果
2021/11/11 HTML / CSS