Tkinter窗口未在辅助显示器上以缩放状态填充屏幕



我正在Windows(10)上构建一个小型python 3 tkinter程序。从主窗口(根)我正在创建一个辅助窗口(wdowViewer)。我希望它在辅助显示器上全屏(放大)。下面的代码适用于我的主设置,有两个相同的屏幕。但是,如果我将笔记本电脑从扩展坞中取出并将其连接到(任何)外部显示器,则新窗口仅填充辅助显示器的约2/3。

需要注意的两件事: - 笔记本电脑和外接显示器具有相同的分辨率。 - 当覆盖重定向设置为 0 时,窗口将适当缩放。

mon_primary_width = str(app.root.winfo_screenwidth()) # width of primary screen
self.wdowViewer = Toplevel(app.root) # create new window
self.wdowViewer.geometry('10x10+' + mon_primary_width + '+0') # move it to the secondary screen
self.wdowViewer.wm_state('zoomed') # full screen
self.wdowViewer.overrideredirect(1) # remove tool bar
app.root.update_idletasks()  # Apply changes

经过两天的实验,我终于找到了解决方法。

请考虑以下示例:使用以下代码时,使顶层缩放在任何辅助显示器上都可以正常工作:

from tkinter import *    
# Make tkinter window
root = Tk()
sw = str(root.winfo_screenwidth())
Label(root, text="Hello Main Display").pack()
# Make a new toplevel
w = Toplevel(root)
w.geometry("0x0+" + sw + "+0")
w.state('zoomed')
w.overrideredirect(1)
Label(w, text='Hello Secondary Display').pack()
root.mainloop()

但是,在我的代码中,我在运行主循环命令后创建了一个新的顶级。然后,问题出现了。存在该问题的代码示例:

from tkinter import *
# New Tkinter
root = Tk()
sw = str(root.winfo_screenwidth())
# Function for new toplevel
def new_wdow():
w = Toplevel(root)
w.geometry("0x0+" + sw + "+0")
w.state('zoomed')
w.overrideredirect(1)
Label(w, text='Hello Secondary Display').pack()
# Make button in main window
Button(root, text="Hello", command=new_wdow).pack()
root.mainloop()

问题:仅当 Windows 中的 DPI 缩放未设置为 100% 时,才会出现此错误。因此,tkinter 在运行主循环后如何处理 DPI 缩放似乎存在错误。它不能正确缩放窗口,但 Windows 会将其视为可以缩放窗口。

解决方法:告诉 Windows,应用会自行处理 DPI 缩放,并且不调整窗口大小。这可以使用 ctypes 实现。将以下代码放在 python 脚本的前面:

import ctypes
ctypes.windll.shcore.SetProcessDpiAwareness(2)

希望其他人会发现该解决方案有所帮助。希望有人可以解释更多这里发生的事情!

最新更新