【问题标题】:Python win32gui SetAsForegroundWindow function not working properlyPython win32gui SetAsForegroundWindow 函数无法正常工作
【发布时间】:2015-05-12 20:01:47
【问题描述】:

我正在尝试编写一个通过搜索窗口标题来查找窗口的程序。一旦它找到了窗口,它就会尝试把它带到前面。我正在使用win32gui API 来实现这一点。我能够让它在大多数情况下工作,但由于某种原因,如果任务管理器在前面,它就不起作用。我有以下示例代码。

import win32gui, win32con
import re, traceback
from time import sleep

class cWindow:
    def __init__(self):
        self._hwnd = None

    def BringToTop(self):
        win32gui.BringWindowToTop(self._hwnd)

    def SetAsForegroundWindow(self):
        win32gui.SetForegroundWindow(self._hwnd)

    def Maximize(self):
        win32gui.ShowWindow(self._hwnd, win32con.SW_MAXIMIZE)

    def setActWin(self):
        win32gui.SetActiveWindow(self._hwnd)

    def _window_enum_callback(self, hwnd, wildcard):
        '''Pass to win32gui.EnumWindows() to check all the opened windows'''
        if re.match(wildcard, str(win32gui.GetWindowText(hwnd))) != None:
            self._hwnd = hwnd

    def find_window_wildcard(self, wildcard):
        self._hwnd = None
        win32gui.EnumWindows(self._window_enum_callback, wildcard)


def main():
    sleep(5)
    try:      
        wildcard = ".*Building Operation WorkStation.*"
        cW = cWindow()
        cW.find_window_wildcard(wildcard)
        cW.Maximize()
        cW.BringToTop()
        cW.SetAsForegroundWindow()

    except:
        f = open("log.txt", "w")
        f.write(traceback.format_exc())
        print traceback.format_exc()
main()

我从多个在线资源拼凑而成。它似乎在大多数情况下都可以工作,但是对于任务管理器之类的某些窗口,它有时会起作用,但其余部分会失败。当它不能正常工作时,我只注意到应用程序图标呈黄色闪烁。是否有适当的方法来确保我感兴趣的窗口 100% 设置为前景?我不确定这是否相关,但我使用的是带有 Service Pack 1 的 Windows 7 Professional(32 位)。

【问题讨论】:

    标签: python windows win32gui setforegroundwindow


    【解决方案1】:

    我找到了一个解决方案:如果是 taskmanager,则将其杀死。我给cWindow添加了一个方法:

    def kill_task_manager(self):
        # Here I use your method to find a window because of an accent in my french OS,
        # but you should use win32gui.FindWindow(None, 'Task Manager complete name').
        wildcard = 'Gestionnaire des t.+ches de Windows'
        self.find_window_wildcard(wildcard)
        if self._hwnd:
            win32gui.PostMessage(self._hwnd, win32con.WM_CLOSE, 0, 0)  # kill it
            sleep(0.5)  # important to let time for the window to be closed
    

    cW = cWindow()之后调用这个方法。

    另一个错误陷阱是防止SetAsForegroundWindow中出现此异常:

    error: (0, 'SetForegroundWindow', 'No error message is available')
    

    在 win32gui 调用之前发送一个 alt 键:

    # Add this import
    import win32com.client
    
    # Add this to __ini__
    self.shell = win32com.client.Dispatch("WScript.Shell")
    
    # And SetAsForegroundWindow becomes
    def SetAsForegroundWindow(self):
        self.shell.SendKeys('%')
        win32gui.SetForegroundWindow(self._hwnd)
    

    最后,如果可以的话,不要比较!= None,而是比较is not None。更多pythonic ;)

    这是完整的代码:

    # coding: utf-8
    
    import re, traceback
    import win32gui, win32con, win32com.client
    from time import sleep
    
    
    class cWindow:
        def __init__(self):
            self._hwnd = None
            self.shell = win32com.client.Dispatch("WScript.Shell")
    
        def BringToTop(self):
            win32gui.BringWindowToTop(self._hwnd)
    
        def SetAsForegroundWindow(self):
            self.shell.SendKeys('%')
            win32gui.SetForegroundWindow(self._hwnd)
    
        def Maximize(self):
            win32gui.ShowWindow(self._hwnd, win32con.SW_MAXIMIZE)
    
        def setActWin(self):
            win32gui.SetActiveWindow(self._hwnd)
    
        def _window_enum_callback(self, hwnd, wildcard):
            '''Pass to win32gui.EnumWindows() to check all the opened windows'''
            if re.match(wildcard, str(win32gui.GetWindowText(hwnd))) is not None:
                self._hwnd = hwnd
    
        def find_window_wildcard(self, wildcard):
            self._hwnd = None
            win32gui.EnumWindows(self._window_enum_callback, wildcard)
    
        def kill_task_manager(self):
            wildcard = 'Gestionnaire des t.+ches de Windows'
            self.find_window_wildcard(wildcard)
            if self._hwnd:
                win32gui.PostMessage(self._hwnd, win32con.WM_CLOSE, 0, 0)
                sleep(0.5)
    
    def main():
        sleep(5)
        try:
            wildcard = ".*Building Operation WorkStation.*"
            cW = cWindow()
            cW.kill_task_manager()
            cW.find_window_wildcard(wildcard)
            cW.BringToTop()
            cW.Maximize()
            cW.SetAsForegroundWindow()
    
        except:
            f = open("log.txt", "w")
            f.write(traceback.format_exc())
            print(traceback.format_exc())
    
    
    if __name__ == '__main__':
        main()
    

    来源:how do I close window with handle using win32gui in Pythonwin32gui.SetActiveWindow() ERROR : The specified procedure could not be found

    【讨论】:

      【解决方案2】:

      注意:以下仅涉及确保在激活窗口之前隐藏任务管理器等始终处于顶部的窗口 - 它假设激活部分本身工作正常,可能并非如此。列出了允许进程调用SetForegroundWindow Windows API 函数的条件here


      任务管理器有两个特别之处:

      • 默认情况下,它设置为始终显示在顶部,即高于所有其他窗口。
      • 即使关闭(Options > Always on Top未选中),您也可以仍然使其显示在其他之上- 顶部窗口(普通窗口似乎无法做到的事情)。

      您的代码:

      • 正在工作 - 在我的测试中 - 在目标窗口确实成为活动窗口的意义上。
      • 在任务管理器窗口仍然停留在(最大化)窗口顶部的意义上工作。
        • 不幸的是,即使尝试将 您的 窗口也设为始终位于顶部的窗口也无济于事。

      专门检查是否存在任务管理器窗口并将其最小化是一个选项,但请注意,可能有其他始终在顶部的窗口,因此您需要一个可靠的解决方案必须识别所有打开的始终在顶部的窗口并最小化它们

      以下尝试识别除任务栏和开始按钮之外的所有始终在顶部的窗口,并最小化(有效隐藏)任何此类窗口。

      新方法是hide_always_on_top_windows_window_enum_callback_hide

      import win32gui, win32con
      import re, traceback
      from time import sleep
      
      class cWindow:
          def __init__(self):
              self._hwnd = None
      
          def SetAsForegroundWindow(self):
              # First, make sure all (other) always-on-top windows are hidden.
              self.hide_always_on_top_windows() 
              win32gui.SetForegroundWindow(self._hwnd)
      
          def Maximize(self):
              win32gui.ShowWindow(self._hwnd, win32con.SW_MAXIMIZE)
      
          def _window_enum_callback(self, hwnd, regex):
              '''Pass to win32gui.EnumWindows() to check all open windows'''
              if self._hwnd is None and re.match(regex, str(win32gui.GetWindowText(hwnd))) is not None:
                  self._hwnd = hwnd
      
          def find_window_regex(self, regex):
              self._hwnd = None
              win32gui.EnumWindows(self._window_enum_callback, regex)
      
          def hide_always_on_top_windows(self):
              win32gui.EnumWindows(self._window_enum_callback_hide, None)
      
          def _window_enum_callback_hide(self, hwnd, unused):
              if hwnd != self._hwnd: # ignore self
                  # Is the window visible and marked as an always-on-top (topmost) window?
                  if win32gui.IsWindowVisible(hwnd) and win32gui.GetWindowLong(hwnd, win32con.GWL_EXSTYLE) & win32con.WS_EX_TOPMOST:
                      # Ignore windows of class 'Button' (the Start button overlay) and
                      # 'Shell_TrayWnd' (the Task Bar).
                      className = win32gui.GetClassName(hwnd)
                      if not (className == 'Button' or className == 'Shell_TrayWnd'):
                          # Force-minimize the window.
                          # Fortunately, this seems to work even with windows that
                          # have no Minimize button.
                          # Note that if we tried to hide the window with SW_HIDE,
                          # it would disappear from the Task Bar as well.
                          win32gui.ShowWindow(hwnd, win32con.SW_FORCEMINIMIZE)
      
      def main():
          sleep(5)
          try:      
              regex = ".*Building Operation WorkStation.*"
              cW = cWindow()
              cW.find_window_regex(regex)
              cW.Maximize()
              cW.SetAsForegroundWindow()
      
          except:
              f = open("log.txt", "w")
              f.write(traceback.format_exc())
              print(traceback.format_exc())
      main()
      

      【讨论】:

      • 不应该这样:win32gui.GetWindowLong(hwnd, win32con.GWL_EXSTYLE) & win32con.WS_EX_TOPMOSTwin32gui.GetWindowLong(hwnd, win32con.GWL_EXSTYLE & win32con.WS_EX_TOPMOST) 吗?你想把它们咬在一起对吗?
      • @IronManMark20: win32gui.GetWindowLong(hwnd, win32con.GWL_EXSTYLE) 返回一个位字段,& win32con.WS_EX_TOPMOST 测试是否设置了特定位。您所做的可能会导致将无效标志传递给GetWindowLong
      猜你喜欢
      • 2019-03-05
      • 2012-02-04
      • 1970-01-01
      • 1970-01-01
      • 2018-07-23
      • 2019-09-13
      • 1970-01-01
      • 1970-01-01
      • 2017-12-15
      相关资源
      最近更新 更多