Открыть окно выбора принтера в Tkinter - PullRequest
0 голосов
/ 04 марта 2020

У меня проблема, возможно, из-за недостатка знаний. Я хочу открыть диалоговое окно windows, чтобы выбрать принтер и отправить его на печать (на принтер) с помощью Tkinter.

В настоящее время я использую код, чтобы связать Tkinter с Wx python и сделать его асинхронным, создавая отдельный процесс.

Это код, упомянутый выше:

from tkinter import *
from threading import Thread
import wx

def f_imprimir(ventana, entry):
    class TextDocPrintout(wx.Printout):
        def __init__(self):
            wx.Printout.__init__(self)

        def OnPrintPage(self, page):
            dc = self.GetDC()

            ppiPrinterX, ppiPrinterY = self.GetPPIPrinter()     
            ppiScreenX, ppiScreenY = self.GetPPIScreen()     
            logScale = float(ppiPrinterX)/float(ppiScreenX)

            pw, ph = self.GetPageSizePixels()
            dw, dh = dc.GetSize()     
            scale = logScale * float(dw)/float(pw)
            dc.SetUserScale(scale, scale)

            logUnitsMM = float(ppiPrinterX)/(logScale*25.4)

            ### Print code ###

            return True

    class PrintFrameworkSample(wx.Frame):        
        def OnPrint(self):
            pdata = wx.PrintData()
            pdata.SetPaperId(wx.PAPER_A4)
            pdata.SetOrientation(wx.LANDSCAPE)

            data = wx.PrintDialogData(pdata)
            printer = wx.Printer(data)

            printout = TextDocPrintout()

            useSetupDialog = True

            if not printer.Print(self, printout, useSetupDialog) and printer.GetLastError() == 
wx.PRINTER_ERROR:

                wx.MessageBox(

                    "There was a problem printing.\n"

                    "Perhaps your current printer is not set correctly?",

                    "Printing Error", wx.OK)

            else:
                data = printer.GetPrintDialogData() 
                pdata = wx.PrintData(data.GetPrintData()) # force a copy

            printout.Destroy()
            self.Destroy()

    app=wx.App(False)
    PrintFrameworkSample().OnPrint()
    entry.config(state="normal")


def process(ventana, entry):
    entry.config(state="disable")
    t = Thread(target=f_imprimir, args=(ventana,entry))
    t.start()

v = Tk()
entry = Entry(v)
entry.pack()
v.bind("a", lambda a:process(v,entry))

после завершения wx.app, что может произойти при закрытии селектора принтеров, я планирую изменить статус записи на "нормальный". Но он выдает ошибку при изменении состояния записи на «нормальное», что, я полагаю, связано с тем, что окно и заказ, который я отправляю, находятся в отдельных процессах. Ошибка будет:

Exception in thread Thread-1:
Traceback (most recent call last):
  File "C:\Python38-32\lib\threading.py", line 932, in _bootstrap_inner
    self.run()
  File "C:\Python38-32\lib\threading.py", line 870, in run
    self._target(*self._args, **self._kwargs)
  File "C:\Users\DANTE\Google Drive\JNAAB\DESARROLLO\pruebas\pedasito.py", line 65, in f_imprimir
    entry.config(state="normal")
  File "C:\Python38-32\lib\tkinter\__init__.py", line 1637, in configure
    return self._configure('configure', cnf, kw)
  File "C:\Python38-32\lib\tkinter\__init__.py", line 1627, in _configure
    self.tk.call(_flatten((self._w, cmd)) + self._options(cnf))
RuntimeError: main thread is not in main loop

У кого-нибудь есть решение этой проблемы или альтернативный способ создания окна печати без блокировки окна TCL и возможности блокировать запись? Если есть способ сделать это или отправить на печать с помощью Tkinter и избежать этого беспорядка, было бы еще лучше. Спасибо.

1 Ответ

1 голос
/ 05 марта 2020

Я не использую wx Python, однако ваша ошибка связана с проблемой многопоточности, связанной с tkinter. Tkinter любит находиться в главном потоке, и попытка передать виджеты в отдельный поток может вызвать проблемы. Однако ваше поле ввода уже находится в глобальном пространстве имен, поэтому вам не нужно его передавать.

Просто обновите его из своей ветки, как только это будет необходимо.

Я бы сделал это в вашем if/else условие, так что это происходит только в правильное время.

Примерно так будет работать: заметьте, что вам нужно будет что-то сделать с переданным значением. Поскольку сейчас ни один из ваших кодов на самом деле не печатает ничего, кроме пустой страницы.

import tkinter as tk
from threading import Thread
import wx


def f_imprimir(value):
    # here you can see the value of entry was passed as a string so we can avoid any issues with the widget
    print(value)
    class TextDocPrintout(wx.Printout):
        def __init__(self):
            wx.Printout.__init__(self)

        def OnPrintPage(self, page):
            dc = self.GetDC()
            ppiPrinterX, ppiPrinterY = self.GetPPIPrinter()     
            ppiScreenX, ppiScreenY = self.GetPPIScreen()     
            logScale = float(ppiPrinterX)/float(ppiScreenX)

            pw, ph = self.GetPageSizePixels()
            dw, dh = dc.GetSize()     
            scale = logScale * float(dw)/float(pw)
            dc.SetUserScale(scale, scale)
            logUnitsMM = float(ppiPrinterX)/(logScale*25.4)
            return True

    class PrintFrameworkSample(wx.Frame):        
        def OnPrint(self):
            pdata = wx.PrintData()
            pdata.SetPaperId(wx.PAPER_A4)
            pdata.SetOrientation(wx.LANDSCAPE)
            data = wx.PrintDialogData(pdata)
            printer = wx.Printer(data)
            printout = TextDocPrintout()
            useSetupDialog = True

            if not printer.Print(self, printout, useSetupDialog) and printer.GetLastError() == wx.PRINTER_ERROR:

                wx.MessageBox("There was a problem printing.\n\n"
                              "Perhaps your current printer is not set correctly?\n\n"
                              "Printing Error", wx.OK)
                entry.config(state="normal")
            else:
                data = printer.GetPrintDialogData() 
                pdata = wx.PrintData(data.GetPrintData())  # force a copy
                entry.config(state="normal")

            printout.Destroy()
            self.Destroy()

    app = wx.App(False)
    PrintFrameworkSample().OnPrint()



def process(_=None):
    entry.config(state="disable")
    t = Thread(target=f_imprimir, args=(entry.get(),))
    t.start()


v = tk.Tk()
entry = tk.Entry(v)
entry.pack()
v.bind("<Return>", process)
v.mainloop()
...