我正在实施一个任务栏replace,docker式应用程序切换器风格的程序。 它使用OpenGL做了一些独特的东西,还有键盘快捷键,所以它的设置方式,窗口并不总是有焦点。 我想实现它,这样我就可以把一个任意窗口放到前台,就像任务栏或ALT-TAB程序一样。
但是,我的代码只是简单地导致应用程序图标在任务栏中闪烁。 Windows API文档说,这是应该发生,但我正在寻找一种方法来解决这个问题。
我已经调整了我的代码从下面的例子,它说附加到前台线程应允许您设置前景窗口。 这里是网站:
http://www.voidnish.com/Articles/ShowArticle.aspx?code=dlgboxtricks
http://invers2008.blogspot.com/2008/10/mfc-how-to-steal-focus-on-2kxp.html
我的代码看起来像这样。 请注意,它是使用Win32的包装的Python(self.hwnd是我想要把前面的窗口的句柄):
fgwin = win32gui.GetForegroundWindow() fg = win32process.GetWindowThreadProcessId(fgwin)[0] current = win32api.GetCurrentThreadId() if current != fg: win32process.AttachThreadInput(fg, current, True) win32gui.SetForegroundWindow(self.hwnd) win32process.AttachThreadInput(fg, win32api.GetCurrentThreadId(), False)
但是,除非我的窗口是前景窗口(通常不是这样),否则这只会导致程序的图标闪烁。
我在做线程连接错误? 有没有其他的方法来解决这个问题? 我认为必须有,因为有很多应用程序切换器,似乎能够做到这一点很好。
我正在用python写这个,但是如果有另一种语言的解决scheme,我将使用包装器或者做任何必要的操作来启动和运行。
提前致谢!
编辑:我会打开一个方法,使其工作只在我的特定的计算机,即一种方式来启用,在我的机器上,任何应用程序的重点。
解决:你必须做的是禁用前台锁。 结果如此简单:
win32gui.SystemParametersInfo(win32con.SPI_SETFOREGROUNDLOCKTIMEOUT, 0, win32con.SPIF_SENDWININICHANGE | win32con.SPIF_UPDATEINIFILE)
我已经有一些已经运行多年的代码,一直回到Windows 95.当双击应用程序系统托盘图标时,我总是使用Win32 API函数(如BringWindowToTop和SetForegroundWindow)将我的应用程序窗口置于前台。 这一切都停止按照预期在Windows 7上工作,其中我的输入窗口将最终落后于其他窗口,窗口图标会闪烁状态栏上。 我想到的“解决办法”是这样的; 它似乎可以在所有版本的Windows上运行。
//-- show the window as you normally would, and bring window to foreground. // for example; ::ShowWindow(hWnd,SW_SHOW); ::BringWindowToTop(hWnd); ::SetForegroundWindow(hWnd); //-- on Windows 7, this workaround brings window to top ::SetWindowPos(hWnd,HWND_NOTOPMOST,0,0,0,0, SWP_NOMOVE | SWP_NOSIZE); ::SetWindowPos(hWnd,HWND_TOPMOST,0,0,0,0,SWP_NOMOVE | SWP_NOSIZE); ::SetWindowPos(hWnd,HWND_NOTOPMOST,0,0,0,0,SWP_SHOWWINDOW | SWP_NOMOVE | SWP_NOSIZE);
我不喜欢这些使用win32gui
建议,因为你不能通过pip
轻松安装。 所以这是我的解决方案:
首先,通过pip
安装pywinauto
。 如果你使用的是Python 2.7.9或者2分支上的更新版本,或者Python 3.4.0或者3分支上的更新版本,那么已经安装了pip
。 对于其他人,请更新Python以获取它(或者,如果您必须运行较旧版本的Python,则可以通过运行此脚本来手动下载并安装它。)
只需从命令行运行(而不是在Python中):
pip install pywinauto
接下来,从pywinauto
导入你需要的东西:
from pywinauto.findwindows import find_window from pywinauto.win32functions import SetForegroundWindow
最后,这只是一个实际的行:
SetForegroundWindow(find_window(title='taskeng.exe'))
根据nspire,我已经尝试了他的解决方案与python 2.7和W8,它的作品就像一个魅力, 即使窗口被最小化 *。
win32gui.ShowWindow(HWND, win32con.SW_RESTORE) win32gui.SetWindowPos(HWND,win32con.HWND_NOTOPMOST, 0, 0, 0, 0, win32con.SWP_NOMOVE + win32con.SWP_NOSIZE) win32gui.SetWindowPos(HWND,win32con.HWND_TOPMOST, 0, 0, 0, 0, win32con.SWP_NOMOVE + win32con.SWP_NOSIZE) win32gui.SetWindowPos(HWND,win32con.HWND_NOTOPMOST, 0, 0, 0, 0, win32con.SWP_SHOWWINDOW + win32con.SWP_NOMOVE + win32con.SWP_NOSIZE)
win32gui.ShowWindow(HWND, win32con.SW_RESTORE)
,它现在适用于所有情况。 如果你正在实现热键,使用RegisterHotKey
。 正如Raymond Chen所说的(同伴博客文章已经与Chris联系在一起), “按一个已注册的热键给你前景激活的爱” 。
SetForegroundWindow
函数的文档解释,这实际上是预期的行为; 流程不应该能够“偷”到重点。 但是,可以调整代码,使其无论如何都能正常工作。
看一下LockSetForegroundWindow
的注释部分:解释
如果用户按下ALT键[…],系统将自动启用对SetForegroundWindow的调用。
在调用SetForegroundWindow
之前,可以通过使程序模拟使用SendInput
函数按Alt键来利用此行为。