这两天为了实现云打印功能找了很多相关的文章
记录一下这一篇,python云打印实现-朝花夕拾,代码通过监听文件夹有无产生新文件来判断是否执行,我尝试运行了下没问题,于是打算转载一下
程序运行结果
由于对方的代码和我实现的有点出入但都是先下载,后打印,只不过我的实通过mqtt服务器罢了,于是通过一步一步解析,我发现他这里有几个需要注意的,给大家和自己才个坑
1、settings.json 配置打印机和监控文件夹
2、不支持图片和pdf打印,有点难弄,后面处理了我会附上
3、不支持彩彩印规格这些
4、打印机用默认的即可,配置没多大作用,因为最后参数没有传过去
我改良后的
import time import win32api import win32print from watchdog.observers import Observer from watchdog.events import * import json import sys import traceback import os def printer_loading(filename, printer): win32api.ShellExecute( 0, "print", filename, '/d:"%s"' % win32print.GetDefaultPrinter(), ".", 0 ) ''' def printer_pic(pic_path, printer_name): # 物理宽度、高度 PHYSICALWIDTH = 110 PHYSICALHEIGHT = 111 # 物理偏移位置 PHYSICALOFFSETX = 112 PHYSICALOFFSETY = 113 printer_name = win32print.GetDefaultPrinter() hDC = win32ui.CreateDC() hDC.CreatePrinterDC(printer_name) printer_size = hDC.GetDeviceCaps(PHYSICALWIDTH), hDC.GetDeviceCaps(PHYSICALHEIGHT) # printer_margins = hDC.GetDeviceCaps (PHYSICALOFFSETX), hDC.GetDeviceCaps (PHYSICALOFFSETY) # 打开图片 # #通过每个像素使它尽可能大 # #页面不失真。 bmp = Image.open(file_name) ratios = [1.0 * 1754 / bmp.size[0], 1.0 * 1240 / bmp.size[1]] scale = min(ratios) # #开始打印作业,并将位图绘制到 # #按比例缩放打印机设备。 hDC.StartDoc(file_name) hDC.StartPage() dib = ImageWin.Dib(bmp) scaled_width, scaled_height = [int(scale * i) for i in bmp.size] print(scaled_width, scaled_height) x1 = int((printer_size[0] - scaled_width) / 2) y1 = int((printer_size[1] - scaled_height) / 2) # 横向位置坐标 x1 = 1580 # 竖向位置坐标 y1 = 30 # 4倍为自适应图片实际尺寸打印 x2 = x1 + bmp.size[0] * 4 y2 = y1 + bmp.size[1] * 4 dib.draw(hDC.GetHandleOutput(), (x1, y1, x2, y2)) hDC.EndPage() hDC.EndDoc() hDC.DeleteDC() ''' class FileEventHandler(FileSystemEventHandler): def __init__(self,printer): FileSystemEventHandler.__init__(self) self.printer = printer def on_created(self, event): print(event.is_directory) if event.is_directory: print("directory created:{0}".format(event.src_path)) else: print("file created:{0}".format(event.src_path)) if event.src_path.endswith("docx") or event.src_path.endswith("doc") or event.src_path.endswith("xlsx") or event.src_path.endswith("jpg") or event.src_path.endswith("pdf") or event.src_path.endswith("xls") or event.src_path.endswith("txt") or event.src_path.endswith("img"): if "~" not in event.src_path: print(f"打印路径{event.src_path}") print(f"打印机名字{self.printer}") printer_loading(event.src_path, self.printer) # import time # win32api.ShellExecute(0, 'open', 'C:\\Program Files (x86)\\Allway Sync\\Bin\\syncappw.exe', '','',1) if __name__ == "__main__": try: observers = [] print("当前检测到的打印机:") maxL = 2 Num = 2 for i in range(1, 10): # print(i) r = win32print.EnumPrinters(i) l = len(r) if l > maxL: maxL = l Num = i for i, p in enumerate(list(win32print.EnumPrinters(Num))): print(f"\t{i}:{p[1]}\n") # f = open('.ttings.txt', 'r', encoding="utf-8") settings = {} # print("当前配置:") # for line in f.readlines(): # k, v = line.split(",") # settings[k.strip()] = v.strip() # print(f"\t{k}:{v}") print("当前配置:") with open("./settings.json", 'rb') as load_f: ds = json.load(load_f) for d in ds: settings[d['url'].strip()] = d["printer"].strip() print(f"\t{d}") pass for path, printer in settings.items(): observer = Observer() print("打印机名"+printer) event_handler = FileEventHandler(printer) observer.schedule(event_handler, path, True) observer.start() observers.append(observer) try: while True: time.sleep(1) except KeyboardInterrupt: for observer in observers: observer.stop() for observer in observers: observer.join() except Exception as e: print(e) traceback.print_exc() input()
完整代码在这里:xxxxx
我顺藤摸瓜发现我们启动的方法不一样,我的是通过
w = DispatchEx('kwps.Application') w.Visible = 0 # 不打开软件 w.DisplayAlerts = 0 # 不报错 doc = w.Documents.Open(f)
而他的核心代码是通过
# 打印输出 def printer_loading(filename, printer): win32api.ShellExecute( 0, "print", filename, '/d:"%s"' % win32print.GetDefaultPrinter(), ".", 0 ) # 打印
暂时只知道第一个代码需要安装wps,第二个未测试
client_mqtt版(需要配合mqtt数据使用,不建议下载,自己用,需要安装wps-珠海zf版本)
client_mqtt_plus(需要配合mqtt数据使用,不建议下载,自己用,需要安装wps-珠海zf版本)
朝花夕拾-改良版(可以下载)