Функция Python win32gui SetAsForegroundWindow не работает должным образом

Я пытаюсь написать программу, которая находит окно по его заголовку. Как только он найдет окно, он попытается вывести его на передний план. Для этого я использую 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% случаев? Я не уверен, что это актуально, но я использую Windows 7 Professional (32-разрядную версию) с пакетом обновления 1.


person Aasam Tasaddaq    schedule 12.05.2015    source источник


Ответы (2)


Нашел решение: если таскменеджер, то убить его. Я добавил метод к 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')

просто отправьте клавишу alt перед вызовом win32gui:

# 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. Более питонический ;)

Это полный код:

# 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()

Источники: как закрыть окно с дескриптором, используя win32gui в Python и win32gui.SetActiveWindow( ) ОШИБКА: указанная процедура не найдена.

person Tiger-222    schedule 18.05.2015

Примечание. Следующее относится только к тому, чтобы убедиться, что всегда открытые окна, такие как диспетчер задач, скрыты перед активацией окна — это предполагает, что сама часть активации работает нормально , что может быть не так. Условия, при которых процесс может вызывать функцию SetForegroundWindow Windows API, перечислены здесь.


Диспетчер задач особенный в двух отношениях:

  • По умолчанию установлено отображение всегда сверху, т. е. над всеми остальными окнами.
  • Даже когда это отключено (Options > Always on Top не отмечено), вы можете по-прежнему отображать его поверх других всегда поверх окон (что-то что обычные окна, казалось бы, не могут сделать).

Ваш код:

  • работает - в моих тестах - в том смысле, что целевое окно становится активным окном.
  • is not working in the sense that the Task Manager window still stays on top of the (maximized) window.
    • even trying to make your window an always-on-top window as well wouldn't help, unfortunately.

В частности, можно проверить наличие окна диспетчера задач и свернуть его, но обратите внимание, что могут быть другие всегда открытые окна, поэтому для надежного решения необходимо определить все открыть окна, которые всегда находятся поверх остальных, и свернуть их:

Следующий код изо всех сил старается идентифицировать все окна, которые всегда находятся сверху, кроме панели задач и кнопки «Пуск», и сворачивает (фактически скрывает) любые такие окна.

Новые методы — 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()
person mklement0    schedule 19.05.2015
comment
Разве это не должно: win32gui.GetWindowLong(hwnd, win32con.GWL_EXSTYLE) & win32con.WS_EX_TOPMOST быть win32gui.GetWindowLong(hwnd, win32con.GWL_EXSTYLE & win32con.WS_EX_TOPMOST) ? Ты пытаешься укусить их вместе, верно? - person IronManMark20; 22.05.2015
comment
@IronManMark20: win32gui.GetWindowLong(hwnd, win32con.GWL_EXSTYLE) возвращает битовое поле, а & win32con.WS_EX_TOPMOST проверяет, установлен ли определенный бит. То, что вы делаете, скорее всего, приведет к передаче недопустимого флага в GetWindowLong. - person mklement0; 22.05.2015