想实现用win32 API在tkinter窗口上画图,那么应该先获得tkinter窗口的句柄hwnd,然后再获得tkinter的设备hdc。尝试了FindWindow(),GetActiveWindow(),GetForegroundWindow()这几个函数貌似都不能正确获得tkinter的句柄
后来在stackoverflow上看到一个列子,使用pywintypes.HANDLE(int(root.frame(), 16))才成功。
原例:https://stackoverflow.com/questions/25602986/using-python-tkinter-how-to-smoothly-move-and-resize-the-non-focusable-window
这里我把原例中创建一个non focusable window的tkinter窗口,修改为把窗口设置样式设置为WS_POPUP
import tkinter as tk import win32api import win32con import pywintypes root = tk.Tk() root.geometry("+0+0") root.lift() hWindow = pywintypes.HANDLE(int(root.frame(), 16)) win32api.SetWindowLong(hWindow,win32con.GWL_STYLE,win32con.WS_POPUP) root.mainloop()
这个窗口风格,类似overrideredirect(1)
我想实现的功能是在tkinter窗口上绘画,那么似乎要考虑在哪里开始使用绘制代码,以及窗口重绘的问题。事实上,我只有把画图代码定义成一个事件的回调函数时,图形才绘制出来。而对窗口重绘没有找到办法
from win32gui import * import win32con import pywintypes from tkinter import * root=Tk() root.title("tkinterApp") hwnd = pywintypes.HANDLE(int(root.frame(), 16)) print(hwnd) hdc=GetDC(hwnd) def callback(): hbrush=GetStockObject(win32con.NULL_BRUSH) oldbrush=SelectObject(hdc,hbrush) Rectangle(hdc,50,50,100,100) Button(root,text="显示画图",command=callback).pack() # #ReleaseDC(hwnd,hdc) root.geometry("400x400") #root.mainloop()
解决方法
from win32gui import * import win32con import pywintypes from tkinter import * root=Tk() hwnd = pywintypes.HANDLE(int(root.frame(), 16)) hdc=GetDC(hwnd) def callback(event): root.update() hbrush=GetStockObject(win32con.NULL_BRUSH) oldbrush=SelectObject(hdc,hbrush) Rectangle(hdc,50,50,100,100) root.geometry("400x400") root.bind("<Configure>",callback) root.mainloop()