經過幾小時的調整,我已經解決了這個代碼,這讓我能夠解決熟悉的模糊/當在Python 3中使用Tkinter接口時,Windows 10中的模糊文本顯示在高DPI上。嘗試在Windows 10高DPI顯示器上解決模糊的tkinter文本+縮放問題,但是擔心我的方法不是Pythonic或不安全
我不想設置兼容性標誌或期望其他人這樣做,我發現通過標記DPI意識'on'通過DLL調用,然後檢索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))
這是完美的,謝謝!這些信息需要更廣泛的瞭解 - 幾個月來一直在尋找這種解決方案。 – FoxDot