尝试在 Windows 10 高 DPI 显示器上解决模糊的 tkinter 文本 + 缩放,但担心我的方法不是 Pythonic 或不安全
Posted
技术标签:
【中文标题】尝试在 Windows 10 高 DPI 显示器上解决模糊的 tkinter 文本 + 缩放,但担心我的方法不是 Pythonic 或不安全【英文标题】:Attempting to resolve blurred tkinter text + scaling on Windows 10 high DPI displays, but concerned my approach is not Pythonic or is unsafe 【发布时间】:2017-05-10 00:16:00 【问题描述】:经过数小时的调整,我已经确定了这段代码,这让我在 Python 3 中使用 Tkinter 接口时,可以解决 Windows 10 中在高 DPI 显示器上出现模糊/模糊文本的常见问题。
我不想设置兼容性标志或期望其他人这样做,我发现通过 DLL 调用将 DPI 感知标记为“开启”,然后检索 DPI 设置,我可以扩大 GUI 窗口和框架内。
然而,在将其传递给其他人之前,我想检查我将“GUI”(一个 tkinter.Tk() 实例)传递给主体中的 MakeTkDPIAware 函数并让该函数向其添加自定义属性的方法是一个健康的选择,否则有可能导致 tkinter 实例出现问题。添加的属性随后可以在主体中使用,但可以安全地假设这种情况总是会发生吗?
我已经能够找出这种做法是否为人所知 - 以及它是否令人不悦或设计选择不佳。 (经常在 Python 中,我可以很兴奋地得到一些工作,以至于我当时忘记检查这类问题),所以我希望有人能给我建议。这似乎是“记住”缩放数据的最简洁的方式,而不是创建一个新的全局变量。
如果另一种解决方案更 Pythonic,我很想知道。
import re
def Get_HWND_DPI(window_handle):
#To detect high DPI displays and avoid need to set Windows compatibility flags
import os
if os.name == "nt":
from ctypes import windll, pointer, wintypes
try:
windll.shcore.SetProcessDpiAwareness(1)
except Exception:
pass # this will fail on Windows Server and maybe early Windows
DPI100pc = 96 # DPI 96 is 100% scaling
DPI_type = 0 # MDT_EFFECTIVE_DPI = 0, MDT_ANGULAR_DPI = 1, MDT_RAW_DPI = 2
winH = wintypes.HWND(window_handle)
monitorhandle = windll.user32.MonitorFromWindow(winH, wintypes.DWORD(2)) # MONITOR_DEFAULTTONEAREST = 2
X = wintypes.UINT()
Y = wintypes.UINT()
try:
windll.shcore.GetDpiForMonitor(monitorhandle, DPI_type, pointer(X), pointer(Y))
return X.value, Y.value, (X.value + Y.value) / (2 * DPI100pc)
except Exception:
return 96, 96, 1 # Assume standard Windows DPI & scaling
else:
return None, None, 1 # What to do for other OSs?
def TkGeometryScale(s, cvtfunc):
patt = r"(?P<W>\d+)x(?P<H>\d+)\+(?P<X>\d+)\+(?P<Y>\d+)" # format "WxH+X+Y"
R = re.compile(patt).search(s)
G = str(cvtfunc(R.group("W"))) + "x"
G += str(cvtfunc(R.group("H"))) + "+"
G += str(cvtfunc(R.group("X"))) + "+"
G += str(cvtfunc(R.group("Y")))
return G
def MakeTkDPIAware(TKGUI):
TKGUI.DPI_X, TKGUI.DPI_Y, TKGUI.DPI_scaling = Get_HWND_DPI(TKGUI.winfo_id())
TKGUI.TkScale = lambda v: int(float(v) * TKGUI.DPI_scaling)
TKGUI.TkGeometryScale = lambda s: TkGeometryScale(s, TKGUI.TkScale)
#Example use:
import tkinter
GUI = tkinter.Tk()
MakeTkDPIAware(GUI) # Sets the windows flag + gets adds .DPI_scaling property
GUI.geometry(GUI.TkGeometryScale("600x200+200+100"))
gray = "#cccccc"
DemoFrame = tkinter.Frame(GUI, width=GUI.TkScale(580), height=GUI.TkScale(180), background=gray)
DemoFrame.place(x=GUI.TkScale(10), y=GUI.TkScale(10))
DemoFrame.pack_propagate(False)
LabelText = "Scale = " + str(GUI.DPI_scaling)
DemoLabel = tkinter.Label(DemoFrame, text=LabelText, width=10, height=1)
DemoLabel.pack(pady=GUI.TkScale(70))
【问题讨论】:
【参考方案1】:简单地使用
from ctypes import windll
windll.shcore.SetProcessDpiAwareness(1)
并允许字体大小调整解决了我在 Windows 10 上的问题。
不知道为什么没有人评论或协助。模糊的字体很痛苦,我希望这个线程有一个辩论和这个 Tkinter 滋扰的最佳解决方案..
【讨论】:
这太完美了,谢谢!这些信息需要更广为人知 - 几个月来一直在寻找这个解决方案。 像魔术一样工作!为了完成这个答案,我只想提一下这个必需的附加行“from ctypes import windll” 这个解决方案很完美,但是有什么解决方案可以默认修复它而不是在每个代码中都编写它。【参考方案2】:不完全回答您的问题,但如果您想在 Windows 10(秋季创作者更新)中修复高 dpi 显示器上 tkinter 窗口的模糊,而无需插入 Python 代码行:
找到您的 python.exe
和 pythonw.exe
可执行文件
右击python.exe
并点击“属性”
点击“兼容性”标签
点击“更改高 DPI 设置”按钮
勾选“覆盖高 DPI 缩放行为”并在下拉菜单中选择“应用程序”
必要时重复pythonw.exe
【讨论】:
以上是关于尝试在 Windows 10 高 DPI 显示器上解决模糊的 tkinter 文本 + 缩放,但担心我的方法不是 Pythonic 或不安全的主要内容,如果未能解决你的问题,请参考以下文章