原文链接:https://www.cnblogs.com/zhaotianff/p/18338275
在C#中判断其它应用全屏可以有多种方案。我这里提供两种思路
使用定时器
在定时器中定时判断当前窗口的状态是否是最大化或者宽高是否等于桌面窗口的宽高。
这种方法我没有去尝试,凭个人经验,感觉定时器会消耗一定的系统资源。
但我还是列一下大概的思路
- 创建定时器
- 在定时器中调用GetDesktopWindow Api函数获取桌面窗口句柄
- 在定时器中调用GetForegroundWindow Api函数获取当前前台窗口句柄
- 在定时器调用GetWindowRect函数获取以前两个窗口的大小,然后进行比较
步骤4也可以使用GetWindowPlacement函数,然后判断窗口的状态是否是最大化。
使用事件钩子函数SetWinEventHook
SetWinEventHook函数可以对各种事件进行Hook,详细的事件可以参考下表
常量/值 | 说明 |
---|---|
EVENT_AIA_START/EVENT_AIA_END 0xA000-0xAFFF | 辅助功能互操作性联盟 (AIA) 指定的 WinEvent 常量值的范围,供整个行业使用。 有关详细信息,请参阅 WinEvent ID 的分配。 |
EVENT_MIN/EVENT_MAX 0x00000001-0x7FFFFFFF | 可能的最低和最高事件值。 |
EVENT_OBJECT_ACCELERATORCHANGE 0x8012 | 对象的 KeyboardShortcut 属性 已更改。 服务器应用程序为它们的辅助性对象发送该事件。 |
EVENT_OBJECT_CLOAKED 0x8017 | 在隐藏窗口时发送。 隐藏的窗口仍然存在,但对用户不可见。 |
EVENT_OBJECT_CONTENTSCROLLED 0x8015 | 窗口对象的滚动已结束。 与 EVENT_SYSTEM_SCROLLEND不同,此事件与滚动窗口相关联。 无论滚动是水平滚动还是垂直滚动,只要滚动操作完成,都应发送此事件。 WinEventProc 回调函数的 hwnd 参数描述滚动窗口;idObject 参数OBJID_CLIENT,idChild 参数CHILDID_SELF。 |
EVENT_OBJECT_CREATE 0x8000 | 已创建 对象。 系统为以下用户界面元素发送此事件:插入点、 标题控件、 列表视图控件、 选项卡控件、 工具栏控件、 树视图控件和 窗口 对象。 服务器应用程序为它们的辅助性对象发送该事件。 在为父对象发送事件之前,服务器必须为对象的所有子对象发送事件。 服务器必须确保在父对象发送此事件之前,已完全创建所有子对象并准备好接受来自客户端的 IAccessible 调用。 由于父对象在其子对象之后创建,因此客户端必须确保在调用 IAccessible::get_accParent之前已创建对象的父对象,尤其是在使用上下文中挂钩函数的情况下。 |
EVENT_OBJECT_DEFACTIONCHANGE 0x8011 | 对象的 DefaultAction 属性 已更改。 操作系统为对话框发送该事件。 服务器应用程序为它们的辅助性对象发送该事件。 |
EVENT_OBJECT_DESCRIPTIONCHANGE 0x800D | 对象的 Description 属性 已更改。 服务器应用程序为它们的辅助性对象发送该事件。 |
EVENT_OBJECT_DESTROY 0x8001 | 对象已被销毁。 系统为以下用户界面元素发送此事件:插入点、标题控件、列表视图控件、选项卡控件、工具栏控件、树视图控件和窗口对象。 服务器应用程序为它们的辅助性对象发送该事件。 客户端假定当父对象发送此事件时,对象的所有子级都会被销毁。 收到此事件后,客户端不会调用对象的 IAccessible 属性或方法。 但是,只要由于 COM 规则) ,接口指针 (存在引用计数,接口指针必须保持有效,但 UI 元素可能不再存在。 对接口指针的进一步调用可能会返回失败错误;为了防止这种情况,服务器 会创建代理对象 并监视其生命周期。 |
EVENT_OBJECT_DRAGSTART 0x8021 | 用户开始拖动元素。 WinEventProc 回调函数的 hwnd、idObject 和 idChild 参数标识要拖动的对象。 |
EVENT_OBJECT_DRAGCANCEL 0x8022 | 用户已结束拖动操作,然后再将拖动的元素放在放置目标上。 WinEventProc 回调函数的 hwnd、idObject 和 idChild 参数标识要拖动的对象。 |
EVENT_OBJECT_DRAGCOMPLETE 0x8023 | 用户删除了放置目标上的元素。 WinEventProc 回调函数的 hwnd、idObject 和 idChild 参数标识要拖动的对象。 |
EVENT_OBJECT_DRAGENTER 0x8024 | 用户将元素拖动到放置目标的边界。 WinEventProc 回调函数的 hwnd、idObject 和 idChild 参数标识放置目标。 |
EVENT_OBJECT_DRAGLEAVE 0x8025 | 用户将元素拖出放置目标的边界。 WinEventProc 回调函数的 hwnd、idObject 和 idChild 参数标识放置目标。 |
EVENT_OBJECT_DRAGDROPPED 0x8026 | 用户删除了放置目标上的元素。 WinEventProc 回调函数的 hwnd、idObject 和 idChild 参数标识放置目标。 |
EVENT_OBJECT_END 0x80FF | 最高的对象事件值。 |
EVENT_OBJECT_FOCUS 0x8005 | 对象已接收键盘焦点。 系统为以下用户界面元素发送此事件:列表视图控件、菜单栏、弹出菜单、切换窗口、选项卡控件、树视图控件和窗口对象。 服务器应用程序为它们的辅助性对象发送该事件。 WinEventProc 回调函数的 hwnd 参数标识接收键盘焦点的窗口。 |
EVENT_OBJECT_HELPCHANGE 0x8010 | 对象的 帮助属性 已更改。 服务器应用程序为它们的辅助性对象发送该事件。 |
EVENT_OBJECT_HIDE 0x8003 | 对象已隐藏。 系统为以下用户界面元素发送此事件:插入点和光标。 服务器应用程序为它们的辅助性对象发送该事件。 为父对象生成此事件时,所有子对象都已隐藏。 服务器应用程序不会为子对象发送此事件。 隐藏的对象包括 STATE_SYSTEM_INVISIBLE 标志;shown 对象不包括此标志。 EVENT_OBJECT_HIDE 事件还指示已设置STATE_SYSTEM_INVISIBLE标志。 因此,在这种情况下,服务器不会发送 EVENT_OBJECT_STATECHANGE 事件。 |
EVENT_OBJECT_HOSTEDOBJECTSINVALIDATED 0x8020 | 承载其他可访问对象的窗口已更改托管对象。 客户端可能需要查询主机窗口以发现新的托管对象,尤其是在客户端一直在监视窗口中的事件时。 托管对象是与主机不同的辅助功能框架 (MSAA 或 UI 自动化) 的对象。 托管对象中与主机相同的框架中的更改应随结构更改事件一起处理,例如 MSAA 的EVENT_OBJECT_CREATE 。 有关详细信息,请参阅 winuser.h 中的注释。 |
EVENT_OBJECT_IME_HIDE 0x8028 | IME 窗口已隐藏。 |
EVENT_OBJECT_IME_SHOW 0x8027 | IME 窗口已变为可见。 |
EVENT_OBJECT_IME_CHANGE 0x8029 | 输入法窗口的大小或位置已更改。 |
EVENT_OBJECT_INVOKED 0x8013 | 已调用 对象;例如,用户单击了一个按钮。 此事件受常见控件支持,由 UI 自动化使用。 对于此事件,WinEventProc 回调函数的 hwnd、ID 和 idChild 参数标识所调用的项。 |
EVENT_OBJECT_LIVEREGIONCHANGED 0x8019 | 属于活动区域的对象已更改。 实时区域是应用程序频繁更改和/或异步更改的区域。 |
EVENT_OBJECT_LOCATIONCHANGE 0x800B | 对象已更改位置、形状和大小。 系统为以下用户界面元素发送此事件:插入点和窗口对象。 服务器应用程序为它们的辅助性对象发送该事件。 生成此事件以响应对象层次结构中顶级对象的更改;它不是为对象可能具有的任何子项生成的。 例如,如果用户调整窗口大小,系统会为窗口发送此通知,但不会针对菜单栏、标题栏、滚动条或其他也已更改的对象发送此通知。 当父窗口移动时,系统不会为所有非浮动子窗口发送该事件。 但是,如果应用程序由于调整父窗口的大小而显式调整子窗口的大小,系统将为重设大小的子窗口发送多个事件。 如果对象的 State 属性 设置为 STATE_SYSTEM_FLOATING,则每当对象更改位置时,服务器就会发送 EVENT_OBJECT_LOCATIONCHANGE 。 如果对象不具有此状态,则服务器仅在对象相对于其父级移动时才触发此事件。 对于此事件通知,WinEventProc 回调函数的 idChild 参数标识已更改的子对象。 |
EVENT_OBJECT_NAMECHANGE 0x800C | 对象的 Name 属性 已更改。 系统为以下用户界面元素发送此事件:检查框、光标、列表视图控件、推送按钮、单选按钮、状态栏控件、树视图控件和窗口对象。 服务器应用程序为它们的辅助性对象发送该事件。 |
EVENT_OBJECT_PARENTCHANGE 0x800F | 对象具有新的父对象。 服务器应用程序为它们的辅助性对象发送该事件。 |
EVENT_OBJECT_REORDER 0x8004 | 容器对象已添加、移除其子对象或对其子对象重新排序。 系统为以下用户界面元素发送此事件:标头控件、列表视图控件、工具栏控件和窗口对象。 服务器应用程序在适当的时候为它们的辅助性对象发送该事件。 例如,当子元素的数量或元素的顺序发生更改时,列表视图对象会生成此事件。 当子窗口的 Z 顺序更改时,父窗口也会发送此事件。 |
EVENT_OBJECT_SELECTION 0x8006 | 容器对象中的选定内容已更改。 系统为以下用户界面元素发送此事件:列表视图控件、选项卡控件、树视图控件和窗口对象。 服务器应用程序为它们的辅助性对象发送该事件。 此事件指示单个选择:在以前不包含任何选定子项的容器中选择了子级,或者所选内容已从一个子级更改为另一个子级。 WinEventProc 回调函数的 hwnd 和 idObject 参数描述容器;idChild 参数标识所选的对象。 如果所选子级是同时包含 对象的窗口,则 idChild 参数 OBJID_WINDOW。 |
EVENT_OBJECT_SELECTIONADD 0x8007 | 容器对象中的子项已添加到现有选定内容中。 系统为以下用户界面元素发送此事件:列表框、列表视图控件和树视图控件。 服务器应用程序为它们的辅助性对象发送该事件。 WinEventProc 回调函数的 hwnd 和 idObject 参数描述容器。 idChild 参数是添加到所选内容的子级。 |
EVENT_OBJECT_SELECTIONREMOVE 0x8008 | 容器对象中的项已从所选内容中删除。 系统为以下用户界面元素发送此事件:列表框、列表视图控件和树视图控件。 服务器应用程序为它们的辅助性对象发送该事件。 此事件指示子项已从现有选定内容中删除。 WinEventProc 回调函数的 hwnd 和 idObject 参数描述容器;idChild 参数标识已从所选内容中删除的子级。 |
EVENT_OBJECT_SELECTIONWITHIN 0x8009 | 容器对象中发生了许多选择更改。 系统为列表框发送此事件;服务器应用程序为其可访问的对象发送它。 当控件中的选定项发生重大更改时,将发送此事件。 该事件通知客户端发生了许多选择更改,并且发送该事件而不是多个 EVENT_OBJECT_SELECTIONADD 或 EVENT_OBJECT_SELECTIONREMOVE 事件。 客户端通过调用容器对象的 IAccessible::get_accSelection 方法并枚举所选项来查询所选项。 对于此事件通知,WinEventProc 回调函数的 hwnd 和 idObject 参数描述发生更改的容器。 |
EVENT_OBJECT_SHOW 0x8002 | 显示隐藏的对象。 系统为下列用户界面元素发送此事件:插入符号、光标和窗口对象。 服务器应用程序为它们的辅助性对象发送该事件。 客户端假定当父对象发送此事件时,已显示所有子对象。 因此,服务器应用程序不会为子对象发送此事件。 隐藏的对象包括 STATE_SYSTEM_INVISIBLE 标志;shown 对象不包括此标志。 EVENT_OBJECT_SHOW 事件还指示已清除STATE_SYSTEM_INVISIBLE标志。 因此,在这种情况下,服务器不会发送 EVENT_OBJECT_STATECHANGE 事件。 |
EVENT_OBJECT_STATECHANGE 0x800A | 对象的状态已更改。 系统为以下用户界面元素发送此事件:检查框、组合框、标题控件、推送按钮、单选按钮、滚动条、工具栏控件、树视图控件、向上-向下控件和窗口对象。 服务器应用程序为它们的辅助性对象发送该事件。 例如,单击或释放按钮对象时,或者启用或禁用某个对象时,会发生状态更改。 对于此事件通知,WinEventProc 回调函数的 idChild 参数标识其状态已更改的子对象。 |
EVENT_OBJECT_TEXTEDIT_CONVERSIONTARGETCHANGED 0x8030 | IME 组合中的转换目标已更改。 转换目标是 IME 组合的子集,主动选择作为用户发起的转换的目标。 |
EVENT_OBJECT_TEXTSELECTIONCHANGED 0x8014 | 对象的文本选择已更改。 此事件受常见控件支持,由 UI 自动化使用。 WinEventProc 回调函数的 hwnd、ID 和 idChild 参数描述更新的文本选择中包含的项。 |
EVENT_OBJECT_UNCLOAKED 0x8018 | 在取消隐藏窗口时发送。 隐藏的窗口仍然存在,但对用户不可见。 |
EVENT_OBJECT_VALUECHANGE 0x800E | 对象的 Value 属性 已更改。 系统为包括滚动条和以下控件的用户界面元素发送此事件:编辑、标头、热键、进度栏、滑块和向上。 服务器应用程序为它们的辅助性对象发送该事件。 |
EVENT_OEM_DEFINED_START/EVENT_OEM_DEFINED_END 0x0101-0x01FF | 为 OEM 保留的事件常量值的范围。 有关详细信息,请参阅 WinEvent ID 的分配。 |
EVENT_SYSTEM_ALERT 0x0002 | 已生成警报。 服务器应用程序不应发送此事件。 |
EVENT_SYSTEM_ARRANGMENTPREVIEW 0x8016 | 正在显示预览矩形。 |
EVENT_SYSTEM_CAPTUREEND 0x0009 | 窗口已丢失鼠标捕获。 此事件由系统发送,从不由服务器发送。 |
EVENT_SYSTEM_CAPTURESTART 0x0008 | 窗口已收到鼠标捕获。 此事件由系统发送,从不由服务器发送。 |
EVENT_SYSTEM_CONTEXTHELPEND 0x000D | 窗口已退出上下文相关帮助模式。 系统不一致地发送此事件。 |
EVENT_SYSTEM_CONTEXTHELPSTART 0x000C | 窗口已进入上下文相关帮助模式。 系统不一致地发送此事件。 |
EVENT_SYSTEM_DESKTOPSWITCH 0x0020 | 已切换活动桌面。 |
EVENT_SYSTEM_DIALOGEND 0x0011 | 对话框已关闭。 系统为标准对话框发送此事件;服务器将其发送到自定义对话框。 系统不一致地发送此事件。 |
EVENT_SYSTEM_DIALOGSTART 0x0010 | 已显示一个对话框。 系统为使用资源模板或 Win32 对话框函数创建的标准对话框发送此事件。 服务器为自定义对话框发送此事件,这些对话框是充当对话框但不是以标准方式创建的窗口。 系统不一致地发送此事件。 |
EVENT_SYSTEM_DRAGDROPEND 0x000F | 应用程序将退出拖放模式。 支持拖放操作的应用程序必须发送此事件;系统不发送此事件。 |
EVENT_SYSTEM_DRAGDROPSTART 0x000E | 应用程序将进入拖放模式。 支持拖放操作的应用程序必须发送此事件,因为系统不会发送它。 |
EVENT_SYSTEM_END 0x00FF | 最高的系统事件值。 |
EVENT_SYSTEM_FOREGROUND 0x0003 | 前景窗口已更改。 即使前台窗口已更改为同一线程中的另一个窗口,系统也会发送此事件。 服务器应用程序从不发送该事件。 对于此事件, WinEventProc 回调函数的 hwnd 参数是前台窗口的句柄, idObject 参数 OBJID_WINDOW, idChild 参数 CHILDID_SELF。 |
EVENT_SYSTEM_MENUPOPUPEND 0x0007 | 弹出菜单已关闭。 系统为标准菜单发送此事件;服务器将其发送到自定义菜单。 当弹出菜单关闭时,客户端将收到此消息,然后 接收EVENT_SYSTEM_MENUEND 事件。 系统不一致地发送此事件。 |
EVENT_SYSTEM_MENUPOPUPSTART 0x0006 | 已显示弹出菜单。 系统为标准菜单发送此事件,这些菜单由 HMENU 标识,并使用菜单模板资源或 Win32 菜单函数创建。 服务器为自定义菜单发送此事件,这些菜单是充当菜单但不以标准方式创建的用户界面元素。 系统不一致地发送此事件。 |
EVENT_SYSTEM_MENUEND 0x0005 | 菜单栏中的菜单已关闭。 系统为标准菜单发送此事件;服务器将其发送到自定义菜单。 对于此事件, WinEventProc 回调函数的 hwnd、 idObject 和 idChild 参数引用包含菜单栏的控件或激活上下文菜单的控件。 hwnd 参数是与事件相关的窗口的句柄。 idObject参数OBJID_MENU或OBJID_SYSMENU菜单,或弹出菜单的OBJID_WINDOW。 idChild参数CHILDID_SELF。 |
EVENT_SYSTEM_MENUSTART 0x0004 | 已选择菜单栏上的菜单项。 系统为标准菜单发送此事件,这些菜单由 HMENU 标识,使用菜单模板资源或 Win32 菜单 API 元素创建。 服务器为自定义菜单发送此事件,自定义菜单是充当菜单但不以标准方式创建的用户界面元素。 对于此事件, WinEventProc 回调函数的 hwnd、 idObject 和 idChild 参数引用包含菜单栏的控件或激活上下文菜单的控件。 hwnd 参数是与事件相关的窗口的句柄。 idObject 参数是菜单的OBJID_MENU或OBJID_SYSMENU,或弹出菜单的OBJID_WINDOW。 idChild参数CHILDID_SELF。 系统触发多个 EVENT_SYSTEM_MENUSTART 事件,这些事件并不总是与 EVENT_SYSTEM_MENUEND 事件相对应。 |
EVENT_SYSTEM_MINIMIZEEND 0x0017 | 即将还原窗口对象。 此事件由系统发送,从不由服务器发送。 |
EVENT_SYSTEM_MINIMIZESTART 0x0016 | 窗口对象即将最小化。 此事件由系统发送,从不由服务器发送。 |
EVENT_SYSTEM_MOVESIZEEND 0x000B | 窗口的移动或调整大小已完成。 此事件由系统发送,从不由服务器发送。 |
EVENT_SYSTEM_MOVESIZESTART 0x000A | 正在移动窗口或调整窗口的大小。 此事件由系统发送,从不由服务器发送。 |
EVENT_SYSTEM_SCROLLINGEND 0x0013 | 滚动条上的滚动已经结束。 此事件由系统为标准滚动条控件和附加到窗口的滚动条发送。 服务器为自定义滚动条发送此事件,这些滚动条是充当滚动条但不是以标准方式创建的用户界面元素。 发送到 WinEventProc 回调函数的 idObject参数OBJID_HSCROLL水平滚动条,垂直滚动条OBJID_VSCROLL。 |
EVENT_SYSTEM_SCROLLINGSTART 0x0012 | 滚动条上的滚动已经开始。 系统为标准滚动条控件和附加到窗口的滚动条发送此事件。 服务器为自定义滚动条发送此事件,这些滚动条是充当滚动条但不是以标准方式创建的用户界面元素。 发送到 WinEventProc 回调函数的 idObject 参数对于水平滚动条OBJID_HSCROLL,垂直滚动条OBJID_VSCROLL。 |
EVENT_SYSTEM_SOUND 0x0001 | 已播放声音。 当系统声音(例如菜单声音)被播放时,系统发送此事件,即使没有声音 (,例如,由于缺少声音文件或声音卡) 。 每当自定义 UI 元素生成声音时,服务器都会发送此事件。 对于此事件, WinEventProc 回调函数接收 OBJID_SOUND 值作为 idObject 参数。 |
EVENT_SYSTEM_SWITCHEND 0x0015 | 用户已释放 Alt+TAB。 此事件由系统发送,从不由服务器发送。 WinEventProc 回调函数的 hwnd 参数标识用户已切换到的窗口。 如果用户按下 Alt+TAB 时只有一个应用程序正在运行,则系统会发送此事件,而不发送相应的 EVENT_SYSTEM_SWITCHSTART 事件。 |
EVENT_SYSTEM_SWITCHSTART 0x0014 | 用户已按 Alt+TAB,这会激活切换窗口。 此事件由系统发送,从不由服务器发送。 WinEventProc 回调函数的 hwnd 参数标识用户要切换到的窗口。 如果用户按 Alt+TAB 时只有一个应用程序正在运行,则系统会发送 EVENT_SYSTEM_SWITCHEND 事件,而不发送相应的 EVENT_SYSTEM_SWITCHSTART 事件。 |
EVENT_UIA_EVENTID_START /EVENT_UIA_EVENTID_END 0x4E00-0x4EFF | 为 UI 自动化事件标识符保留的事件常量值的范围。 有关详细信息,请参阅 WinEvent ID 的分配。 |
EVENT_UIA_PROPID_START/EVENT_UIA_PROPID_END 0x7500-0x75FF | 为 UI 自动化属性更改事件标识符保留的事件常量值的范围。 有关详细信息,请参阅 WinEvent ID 的分配。 |
这里我们Hook EVENT_OBJECT_LOCATIONCHANGE事件就可以,它可以在对象已更改位置、形状和大小时通知。
简单介绍一下SetWinEventHook函数,它的声明如下:
1 HWINEVENTHOOK SetWinEventHook( 2 [in] DWORD eventMin, 3 [in] DWORD eventMax, 4 [in] HMODULE hmodWinEventProc, 5 [in] WINEVENTPROC pfnWinEventProc, 6 [in] DWORD idProcess, 7 [in] DWORD idThread, 8 [in] DWORD dwFlags 9 );
各参数说明如下:
eventMin:事件常量的最小值,取值来源可以看前面的表。可以使用EVENT_MIN 常量来代表可用的最小值,因为我们我们这里只需要Hook一种事件,所以直接传EVENT_OBJECT_LOCATIONCHANGE就可以
eventMax:事件常量的最大值
hmodWinEventProc:包含钩子函数的模块,如果是来自DLL,需要指定,如果是来自当前模块,指定为NULL
pfnWinEventProc:事件钩子函数
idProcess:进程ID,指定0时代表所有进程
idThread:线程ID,指定0时代表所有线程
dwFlags:标记值,取值如下
值 | 含义 |
---|---|
WINEVENT_INCONTEXT | 包含回调函数的 DLL 映射到生成事件的进程的地址空间中。 使用此标志,系统会在事件通知发生时向回调函数发送事件通知。 指定此标志时,挂钩函数必须位于 DLL 中。 当调用进程和生成进程都不是 32 位或 64 位进程,或者生成进程是控制台应用程序时,此标志不起作用。 有关详细信息,请参阅 上下文中挂钩函数。 |
WINEVENT_OUTOFCONTEXT | 回调函数不会映射到生成事件的进程的地址空间中。 由于挂钩函数是跨进程边界调用的,因此系统必须对事件进行排队。 虽然此方法是异步的,但事件保证按顺序排列。 有关详细信息,请参阅 上下文外挂钩函数。 |
WINEVENT_SKIPOWNPROCESS | 防止挂钩的此实例接收此进程中线程生成的事件。 此标志不会阻止线程生成事件。 |
WINEVENT_SKIPOWNTHREAD | 防止此挂钩实例接收注册此挂钩的线程生成的事件。 |
在C#中使用步骤如下
1、导入WinAPI函数
1 public class User32 2 { 3 public static readonly uint SPI_SETDESKWALLPAPER = 0x0014; 4 public static readonly uint SPIF_UPDATEINIFILE = 0x0001; 5 6 public static readonly uint EVENT_OBJECT_LOCATIONCHANGE = 0x800B; 7 public static readonly uint WINEVENT_OUTOFCONTEXT = 0x0000; // Events are ASYNC 8 public static readonly uint WINEVENT_SKIPOWNPROCESS = 0x0002; // Don't call back for events on installer's process 9 10 public static readonly uint SW_SHOWMAXIMIZED = 3; 11 12 public delegate void Wineventproc(IntPtr hWinEventHook, uint eventId, IntPtr hwnd, int idObject, int idChild, uint idEventThread, uint dwmsEventTime); 13 14 15 [DllImport("User32.dll", CharSet = CharSet.Auto)] 16 public static extern IntPtr SetWinEventHook(uint eventMin, uint eventMax, IntPtr hmodWinEventProc, Wineventproc pfnWinEventProc, uint idProcess, uint idThread, uint dwFlags); 17 18 [DllImport("User32.dll")] 19 public static extern bool UnhookWinEvent(IntPtr hWinEventHook); 20 21 [DllImport("User32.dll")] 22 public static extern bool GetWindowPlacement(IntPtr hWnd, ref WINDOWPLACEMENT lpwndpl); 23 24 [DllImport("User32.dll")] 25 public static extern IntPtr GetForegroundWindow(); 26 27 [DllImport("User32.dll")] 28 public static extern IntPtr GetDesktopWindow(); 29 30 [DllImport("User32.dll")] 31 public static extern bool GetWindowRect(IntPtr hWnd, ref System.Drawing.Rectangle lpRect); 32 } 33 34 public struct WINDOWPLACEMENT 35 { 36 public uint length; 37 public uint flags; 38 public uint showCmd; 39 public System.Drawing.Point ptMinPosition; 40 public System.Drawing.Point ptMaxPosition; 41 public System.Drawing.Rectangle rcNormalPosition; 42 public System.Drawing.Rectangle rcDevice; 43 };
2、定义事件钩子函数
就是在这个函数里判断其它应用是否切换到全屏状态
1 private void EventProc(IntPtr hWinEventHook, uint eventId, IntPtr hwnd, int idObject, int idChild, uint idEventThread, uint dwmsEventTime) 2 { 3 if (hwnd == IntPtr.Zero) 4 return; 5 6 //判断是否是前台应用 7 //如果不需要判断是否是前台应用,这个判断可以移除 8 if (User32.GetForegroundWindow() != hwnd) 9 return; 10 11 if (User32.EVENT_OBJECT_LOCATIONCHANGE == eventId) 12 { 13 WINDOWPLACEMENT wp = new WINDOWPLACEMENT(); 14 wp.length = (uint)Marshal.SizeOf(wp); 15 if (User32.GetWindowPlacement(hwnd, ref wp)) //Not very accurate 16 { 17 if (User32.SW_SHOWMAXIMIZED == wp.showCmd) 18 { 19 //应用全屏了 20 } 21 else 22 { 23 //应用退出全屏了 24 } 25 } 26 27 } 28 }
3、创建事件钩子
1 private IntPtr g_eventhook; 2 private User32.Wineventproc Wineventproc;
1 Wineventproc = new Wineventproc(EventProc); 2 3 g_eventhook = User32.SetWinEventHook(User32.EVENT_OBJECT_LOCATIONCHANGE, 4 User32.EVENT_OBJECT_LOCATIONCHANGE, 5 IntPtr.Zero, 6 Wineventproc, 0, 0, 7 User32.WINEVENT_OUTOFCONTEXT | User32.WINEVENT_SKIPOWNPROCESS);
钩子创建成功后,就可以在钩子函数中进行处理相关逻辑了
4、释放事件钩子
在程序退出时,需要调用UnhookWinEvent函数释放钩子。
1 User32.UnhookWinEvent(g_eventhook);
5、运行效果
示例代码下载
参考资料
https://stackoverflow.com/questions/17436795/setwineventhook-window-maximized-event/34549801#34549801
https://learn.microsoft.com/en-us/windows/win32/winauto/event-constants
https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-unhookwinevent
https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-setwineventhook
https://stackoverflow.com/questions/7009080/detecting-full-screen-mode-in-windows