Я нашел ответ. Вместо подписки на активированные и деактивированные события обработайте сообщение WM_ACTIVATE (используется как для активации, так и для деактивации) в WndProc. Поскольку он сообщает о дескрипторе активируемого окна, я могу сравнить этот дескриптор с дескрипторами моих форм и определить, изменяется ли фокус на какую-либо из них.
const int WM_ACTIVATE = 0x0006;
const int WA_INACTIVE = 0;
const int WA_ACTIVE = 1;
const int WA_CLICKACTIVE = 2;
protected override void WndProc(ref Message m)
{
if (m.Msg == WM_ACTIVATE)
{
// When m.WParam is WA_INACTIVE, the window is being deactivated and
// m.LParam is the handle of the window that will be activated.
// When m.WParam is WA_ACTIVE or WA_CLICKACTIVE, the window is being
// activated and m.LParam is the handle of the window that has been
// deactivated.
}
base.WndProc(ref m);
}
Редактировать: Этот метод можно использовать за пределами окна, к которому он применяется (например, снаружи во всплывающем окне).
Вы можете использовать NativeWindow для присоединения к любому окну на основе его дескриптора и просмотра его цикла сообщений. Смотрите пример кода ниже:
public class Popup : Form
{
const int WM_ACTIVATE = 0x0006;
const int WA_INACTIVE = 0;
private ParentWindowIntercept parentWindowIntercept;
public Popup(IntPtr hWndParent)
{
this.parentWindowIntercept = new ParentWindowIntercept(hWndParent);
}
private class ParentWindowIntercept : NativeWindow
{
public ParentWindowIntercept(IntPtr hWnd)
{
this.AssignHandle(hWnd);
}
protected override void WndProc(ref Message m)
{
if (m.Msg == WM_ACTIVATE)
{
if ((int)m.WParam == WA_INACTIVE)
{
IntPtr windowFocusGoingTo = m.LParam;
// Compare handles here
}
}
base.WndProc(ref m);
}
}
}