Подтвердить что ты не робот

Обнаружение активного окна с помощью С# без опроса

Как можно вызвать обратный вызов всякий раз, когда изменяется текущее активное окно. Я видел, как это можно сделать с помощью CBTProc. Однако глобальные события нелегко подключить к управляемому коду. Мне интересно найти способ, который не требует опроса. Я бы предпочел подход, основанный на событиях.

Привет

4b9b3361

Ответ 1

Создайте новый проект форм Windows, добавьте текстовое поле, сделайте его многострочным и установите для свойства Dock текстового поля значение, назовите его. Войдите и вставьте следующий код (вам нужно добавить System.Runtime.InteropServices к вашему usings)...

    WinEventDelegate dele = null;

    public Form1()
    {
        InitializeComponent();
        dele = new WinEventDelegate(WinEventProc);
        IntPtr m_hhook = SetWinEventHook(EVENT_SYSTEM_FOREGROUND, EVENT_SYSTEM_FOREGROUND, IntPtr.Zero, dele, 0, 0, WINEVENT_OUTOFCONTEXT);
    }

    delegate void WinEventDelegate(IntPtr hWinEventHook, uint eventType, IntPtr hwnd, int idObject, int idChild, uint dwEventThread, uint dwmsEventTime);

    [DllImport("user32.dll")]
    static extern IntPtr SetWinEventHook(uint eventMin, uint eventMax, IntPtr hmodWinEventProc, WinEventDelegate lpfnWinEventProc, uint idProcess, uint idThread, uint dwFlags);

    private const uint WINEVENT_OUTOFCONTEXT = 0;
    private const uint EVENT_SYSTEM_FOREGROUND = 3;

    [DllImport("user32.dll")]
    static extern IntPtr GetForegroundWindow();

    [DllImport("user32.dll")]
    static extern int GetWindowText(IntPtr hWnd, StringBuilder text, int count);

    private string GetActiveWindowTitle()
    {
        const int nChars = 256;
        IntPtr handle = IntPtr.Zero;
        StringBuilder Buff = new StringBuilder(nChars);
        handle = GetForegroundWindow();

        if (GetWindowText(handle, Buff, nChars) > 0)
        {
            return Buff.ToString();
        }
        return null;
    }

    public void WinEventProc(IntPtr hWinEventHook, uint eventType, IntPtr hwnd, int idObject, int idChild, uint dwEventThread, uint dwmsEventTime)
    {
        Log.Text += GetActiveWindowTitle() + "\r\n";
    } 

Ответ 2

Я знаю, что эта ветка устарела, но для будущего использования: при запуске кода вы через некоторое время заметите сбой. Это вызвано из строки в конструкторе формы:

public Form1()
    {
        InitializeComponent();
        WinEventDelegate dele = new WinEventDelegate(WinEventProc);//<-causing ERROR
        IntPtr m_hhook = SetWinEventHook(EVENT_SYSTEM_FOREGROUND, EVENT_SYSTEM_FOREGROUND, IntPtr.Zero, dele, 0, 0, WINEVENT_OUTOFCONTEXT);
    }

Вместо приведенного выше сделайте следующую модификацию:

public Form1()
        {
            InitializeComponent();
            dele = new WinEventDelegate(WinEventProc); 
            IntPtr m_hhook = SetWinEventHook(EVENT_SYSTEM_FOREGROUND, EVENT_SYSTEM_FOREGROUND, IntPtr.Zero, dele, 0, 0, WINEVENT_OUTOFCONTEXT);
        }
WinEventDelegate dele = null;

.. работает сейчас, как ожидалось!

Ответ 3

Вы можете использовать SetWinEventHook и прослушать событие EVENT_SYSTEM_FOREGROUND. Используйте флаг WINEVENT_OUTOFCONTEXT, чтобы избежать проблемы с глобальным крюком.