我试图在关闭外部应用程序时收到警报。对于我的测试,我检索了记事本的 hwnd 并将其设置到我的显示器中。奇怪的是,每当我从记事本点击到其他应用程序时,即使记事本没有关闭,我也会收到系统事件。有谁知道为什么?
一些快速示例调用代码:
uint EVENT_OBJECT_DESTROY = 0x8001;
IntPtr notepadHwnd = <your pointer>
var mon = new WindowMonitor(notepadHwnd, EVENT_OBJECT_DESTROY);
mon.EventOccurred += (sender, args) => Console.WriteLine("closed");
监视器:
public class WindowMonitor : IDisposable
{
//store delegate to prevent GC
private User32.WinEventDelegate dEvent;
private IntPtr _hook;
private readonly IntPtr _window;
private readonly List<uint> _watchedEvents = new List<uint>();
public event EventHandler<AccessibleEventTypeEventArgs> EventOccurred;
public WindowMonitor(IntPtr windowToMonitor, params User32.AccessibleEventType[] eventsToMonitor)
{
//prevent junk
if (eventsToMonitor == null || eventsToMonitor.Length == 0)
throw new ArgumentNullException("eventsToMonitor", "Must specify events to monitor");
if (windowToMonitor == IntPtr.Zero)
throw new ArgumentNullException("windowToMonitor", "Must specify a valid window handle to monitor");
_window = windowToMonitor;
//cast them now so we dont have to cast each one when evaluting events
uint lowest = (uint) User32.AccessibleEventType.EVENT_MAX;
uint highest = (uint) User32.AccessibleEventType.EVENT_MIN;
foreach (User32.AccessibleEventType eventType in eventsToMonitor)
{
var castType = (uint) eventType;
_watchedEvents.Add(castType);
//need the range of events to subscribe to
if (castType > highest)
highest = castType;
if (castType < lowest)
lowest = castType;
}
//sign up for the event
dEvent = this.winEvent;
_hook = User32.SetWinEventHook(lowest, highest, IntPtr.Zero, dEvent, 0, 0, User32.WINEVENT_OUTOFCONTEXT);
//ensure it worked
if (IntPtr.Zero.Equals(_hook)) throw new Win32Exception();
//no kill callback
GC.KeepAlive(dEvent);
}
private void winEvent(IntPtr hWinEventHook, uint eventType, IntPtr hWnd, int idObject, int idChild, uint dwEventThread, uint dwmsEventTime)
{
//only care about the events for the specified window
if (hWnd != _window)
return;
if (_watchedEvents.Contains(eventType))
{
if (EventOccurred != null)
EventOccurred(this, new AccessibleEventTypeEventArgs((User32.AccessibleEventType) eventType));
}
}
public void Dispose()
{
//unhook the listener
if (!IntPtr.Zero.Equals(_hook))
User32.UnhookWinEvent(_hook);
//clear variables
_hook = IntPtr.Zero;
dEvent = null;
//kill any event listeners
EventOccurred = null;
GC.SuppressFinalize(this);
}
}
和一些winapi(我省略了事件常量列表):
public delegate void WinEventDelegate(IntPtr hWinEventHook, uint eventType, IntPtr hwnd,
int idObject, int idChild, uint dwEventThread, uint dwmsEventTime);
[DllImport("user32.dll")]
public static extern IntPtr SetWinEventHook(uint eventMin, uint eventMax, IntPtr hmodWinEventProc,
WinEventDelegate lpfnWinEventProc, uint idProcess, uint idThread, uint dwFlags);
[DllImport("user32.dll")]
public static extern bool UnhookWinEvent(IntPtr hWinEventHook);