收到打印作业Python

时间:2015-12-29 21:51:23

标签: python windows winapi printing pywin32

在假脱机期间最初在本地计算机上请求打印作业时,我已成功设法触发回调。但是,无论如何使用win32print或类似的东西可以让我处理将打印作业传输到打印服务器或USB打印机的事件?

################################################################################
# Imports ######################################################################
################################################################################

from os.path import *
from printer import *
from watcher import *
from statvar import *

################################################################################
# Event Callback ###############################################################
################################################################################

def callback(code, event):

    num = splitext(event)[0]
    ext = splitext(event)[1]

    if code == 1 and ext == '.SPL':
        main(num.lstrip('0'))

################################################################################
# wx Event Handler #############################################################
################################################################################

def handling(*args):

    wx.CallAfter(callback, *args)

################################################################################
# Create Listener ##############################################################
################################################################################

# listens to the spool directory for files

watch = Watcher(SPOOL_DIRECTORY, handling)

# set the appropriate flags for a listener

watch.flags = FILE_NOTIFY_CHANGE_FILE_NAME

################################################################################
# Start Listener ###############################################################
################################################################################

watch.start()

################################################################################
# Start wx App #################################################################
################################################################################

app = wx.App()
wx.Frame(None)
app.MainLoop()

################################################################################
################################################################################
################################################################################

1 个答案:

答案 0 :(得分:3)

这是一个可以在我的电脑上运行的想法(Windows 8)。它几乎不是完全成熟的代码,但它可能会让你前进。您需要使用函数FindFirstPrinterChangeNotificationFindNextPrinterChangeNotification这些函数包含在客户端winspool.drv内(令人恼火的是,您可以将其记录为spoolSS.dll,但这是服务器端 - this diagram可以澄清)。

可以从MSDN here获取可以收听的事件列表(以及重要的是,它们的标记设置)。最初我认为你想要PRINTER_CHANGE_ADD_JOB0x00000100),但我认为你可能真的想要PRINTER_CHANGE_WRITE_JOB0x00000800)。一旦作业开始假脱机,这不会触发,但不幸的是,在将一个文档发送到网络打印机的示例中,它似乎确实被触发了多次。

不幸的是,这些API未在win32print库中公开。我想,因此你必须深入研究ctypes。在这里我没有注册回调,而是我监听通知,当我触发时,我调用该函数并在无限循环中再次开始监听。这个过程在听的过程中停滞不前。如果你需要传统的回调功能,你可以在自己的线程中运行这个脚本,或者这个答案可能不适合你的需要。

注意 - 这只是监听所请求的打印作业,然后调用一个函数。如果要提取有关正在触发的作业的信息,代码将变得可怕。进一步说明 - 它将触发启动并随后取消的打印作业,但我想这没关系。

from ctypes import *
from ctypes.wintypes import HANDLE, LPSTR

def add_job_callback():
    print('A job has just been sent to the printer this script is monitoring')

spl = windll.LoadLibrary('winspool.drv')

printer_name = 'KONICA MINOLTA PS Color Laser Class Driver'
# Put the name of your printer here - can be networked or any installed on your computer.  Alternatively, set it to None to use the local printer server
#printer_name = None

hPrinter = HANDLE()

if printer_name:
    spl.OpenPrinterA(c_char_p(printer_name), byref(hPrinter),None)
else:
    spl.OpenPrinterA(None, byref(hPrinter),None)

print(hPrinter)


hjob = spl.FindFirstPrinterChangeNotification(hPrinter,0x00000100,0, None)
# 0x00000100 is a flags setting to set watch for only PRINTER_CHANGE_ADD_JOB
while True:
    windll.kernel32.WaitForSingleObject(hjob,-1)
    #When this function returns, the change that you're monitoring for has been observed, trigger the function
    add_job_callback()
    spl.FindNextPrinterChangeNotification(hjob, None, None, None)

请注意,Python 2.7和Python 3之间存在一些细微差别 - 例如从字符串初始化c_char_p ctype。我已经提供了我能在这里使用的最简单的版本 - 它在2.7中工作。

<强>后记

我做了所有繁重的工作,然后找到this answer,这是重复的。它有更好的代码来处理unicode打印机名称等,但只查看默认的本地打印服务器。

相关问题