我正在开发一个Python应用程序,我想获得每个打开的窗口的HWND
。 我需要窗口和HWND
的名称来过滤列表来pipe理一些特定的窗口,移动和调整它们的大小。
我试图自己看看周围的信息,但我没有得到正确的一段代码。 我试着用这个代码,但我只得到每个窗口的标题(这是伟大的),但我也需要HWND
。
import ctypes import win32gui EnumWindows = ctypes.windll.user32.EnumWindows EnumWindowsProc = ctypes.WINFUNCTYPE(ctypes.c_bool, ctypes.POINTER(ctypes.c_int), ctypes.POINTER(ctypes.c_int)) GetWindowText = ctypes.windll.user32.GetWindowTextW GetWindowTextLength = ctypes.windll.user32.GetWindowTextLengthW IsWindowVisible = ctypes.windll.user32.IsWindowVisible titles = [] def foreach_window(hwnd, lParam): if IsWindowVisible(hwnd): length = GetWindowTextLength(hwnd) buff = ctypes.create_unicode_buffer(length + 1) GetWindowText(hwnd, buff, length + 1) titles.append((hwnd, buff.value)) return True EnumWindows(EnumWindowsProc(foreach_window), 0) for i in range(len(titles)): print(titles)[i] win32gui.MoveWindow((titles)[5][0], 0, 0, 760, 500, True)
这里有一个错误:
win32gui.MoveWindow((titles)[5][0], 0, 0, 760, 500, True) TypeError: The object is not a PyHANDLE object
你混了ctypes
和win32gui
。
你得到的hwnd
是通过ctypes
获得的,并且是一个LP_c_long
对象。 这就是为什么win32gui.MoveWindow
不接受它。 你应该把它传递给
ctypes.windll.user32.MoveWindow(titles[5][0], 0, 0, 760, 500, True)
如果你想使用win32gui.MoveWindow
,你可以直接使用python函数作为回调函数。
例如,
import win32gui def enumHandler(hwnd, lParam): if win32gui.IsWindowVisible(hwnd): if 'Stack Overflow' in win32gui.GetWindowText(hwnd): win32gui.MoveWindow(hwnd, 0, 0, 760, 500, True) win32gui.EnumWindows(enumHandler, None)
你的问题(现在马蒂诺已经解决了你最初的问题,根本不存储HWND
值)是你试图混合ctypes
和win32gui
。
如果你知道自己在做什么,你可以这么做 – 但是如果不知道,就不要这样做。
如果你想获得窗口句柄,你可以使用win32gui
,使用win32gui.EnumWindows
而不是调用user32
DLL的原始函数。
只需修改所有标题的代码片段即可:
titles.append((hwnd, buff.value))
titles
列表将成为包含HWND 和标题文本的元组列表。
要获得所有可用的主窗口的句柄,您将0传递给win32gui.EnumChildWindows
然后检查以确保窗口的文本长度大于0(因为您只需要实际的窗口不隐藏/临时/弹出/特殊窗口)。