2011-06-10 74 views
14

我正在实施任务栏替换,码头式应用程序切换程序样式程序。它使用OpenGL做了一些独特的东西,并且使用键盘快捷键,所以它的设置方式,窗口并不总是有焦点。我想实现它,以便我可以将任意窗口放在前台,就像任务栏或ALT-TAB程序一样。Windows 7:无论其他窗口有什么焦点,如何将窗口置于前面?

但是,我的代码只是简单地导致应用程序图标在任务栏中闪烁。 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写这篇文章,但是如果有另一种语言的解决方案,我将使用包装或者做任何必要的工作来完成这个工作。

在此先感谢!

编辑:我很乐意让它在我的特定计算机上工作,即在我的机器上启用任何应用程序关注的方式。

求助:你要做的就是禁用前台锁。原来,这是因为这很容易:

win32gui.SystemParametersInfo(win32con.SPI_SETFOREGROUNDLOCKTIMEOUT, 0, win32con.SPIF_SENDWININICHANGE | win32con.SPIF_UPDATEINIFILE) 
+2

好的,但有很多应用程序能够做到这一点。这显然是可能的,我发现教程告诉我如何。我的问题是,我在做什么导致教程中的代码无法工作? – jmite 2011-06-10 22:28:55

+0

你为什么不从这些教程中选择一些代码并运行它? – 2011-06-10 22:33:29

+1

这就是我所做的。我发布的代码几乎是教程中的确切代码,在python包装中(请参阅我发布的两个链接)。我问是否有人能说出为什么我的代码没有按照页面所说的做,或者有更好的方法来做。 – jmite 2011-06-10 22:39:48

回答

6

我已经有一些代码已经运行多年,一直回到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); 
+0

它将它带到上面,但键盘焦点不在我们的顶部窗口中。 – Noitidart 2016-07-02 08:54:47

1

SetForegroundWindow函数的文档解释说,这其实是预期的行为;流程不应该能够“偷走”焦点。但是,可以调整您的代码,使其无论如何都能正常工作。

看一看的LockSetForegroundWindow此言一节:它解释

系统将自动启用呼叫如果用户按下ALT键SetForegroundWindow [..]

您可以利用此通过在调用SetForegroundWindow之前使程序模拟使用SendInput函数按下Alt键来执行此操作。

5

据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) 
  • 原来是如果窗口它没有最小化,但由于Whome的评论win32gui.ShowWindow(HWND, win32con.SW_RESTORE),现在适用于所有情况。
+2

您必须在调用SetWindowPos函数之前恢复窗口。 Delphi应用程序可以使用“Application.Restore;”函数来取消最小化。寻找ShowWindow(句柄,SW_RESTORE)python调用。即使窗口已经显示,每次只要调用它就不会造成伤害。 – Whome 2013-12-20 16:04:35

9

我不喜欢这些使用win32gui的建议,因为您无法通过pip轻松安装。所以这里是我的解决方案:

首先,通过pip安装pywinauto。如果您使用的是Python 2.7.9或2分支上的较新版本,或者Python 3.4.0或3分支中的较新版本,则已安装pip。对于其他人,更新Python来得到它(或者你可以手动下载并运行this script,如果你必须运行Python的旧版本进行安装。)

只需在命令行中(而不是从内Python)的运行这个:

pip install pywinauto 

下,进口从pywinauto需要的东西:

from pywinauto.findwindows import find_window 
from pywinauto.win32functions import SetForegroundWindow 

最后,它只是一个实际的行:

SetForegroundWindow(find_window(title='taskeng.exe'))