using System; using System.Collections.Generic; using System.Diagnostics; using System.Runtime.InteropServices; using System.Text; using System.Threading; using System.Threading.Tasks; using System.Windows.Threading; namespace Ink_Canvas.Helpers { /// /// 悬浮窗拦截器 - 检测和隐藏指定的悬浮窗 /// public class FloatingWindowInterceptor : IDisposable { #region Windows API Declarations [DllImport("user32.dll")] private static extern bool EnumWindows(EnumWindowsProc enumProc, IntPtr lParam); [DllImport("user32.dll")] private static extern IntPtr GetWindowThreadProcessId(IntPtr hWnd, out uint processId); [DllImport("user32.dll")] private static extern bool IsWindowVisible(IntPtr hWnd); [DllImport("user32.dll")] private static extern int GetWindowText(IntPtr hWnd, StringBuilder lpString, int nMaxCount); [DllImport("user32.dll")] private static extern int GetClassName(IntPtr hWnd, StringBuilder lpClassName, int nMaxCount); [DllImport("user32.dll")] private static extern bool ShowWindow(IntPtr hWnd, int nCmdShow); [DllImport("user32.dll")] private static extern bool IsIconic(IntPtr hWnd); [DllImport("user32.dll")] private static extern IntPtr GetWindow(IntPtr hWnd, uint uCmd); [DllImport("user32.dll")] private static extern uint GetWindowLong(IntPtr hWnd, int nIndex); [DllImport("user32.dll")] private static extern bool SetWindowPos(IntPtr hWnd, IntPtr hWndInsertAfter, int X, int Y, int cx, int cy, uint uFlags); [DllImport("user32.dll")] private static extern bool IsWindow(IntPtr hWnd); [DllImport("user32.dll")] private static extern bool GetWindowRect(IntPtr hWnd, out ForegroundWindowInfo.RECT lpRect); [DllImport("kernel32.dll")] private static extern IntPtr OpenProcess(uint dwDesiredAccess, bool bInheritHandle, uint dwProcessId); [DllImport("kernel32.dll")] private static extern bool CloseHandle(IntPtr hObject); [DllImport("kernel32.dll")] private static extern int GetProcessImageFileName(IntPtr hProcess, StringBuilder lpImageFileName, int nSize); private delegate bool EnumWindowsProc(IntPtr hWnd, IntPtr lParam); private const int SW_HIDE = 0; private const int SW_SHOW = 1; private const int SW_MINIMIZE = 6; private const int SW_RESTORE = 9; private const int GWL_EXSTYLE = -20; private const uint WS_EX_TOOLWINDOW = 0x00000080; private const uint WS_EX_APPWINDOW = 0x00040000; private const uint SWP_NOMOVE = 0x0002; private const uint SWP_NOSIZE = 0x0001; private const uint SWP_NOZORDER = 0x0004; private const uint SWP_HIDEWINDOW = 0x0080; private const uint PROCESS_QUERY_INFORMATION = 0x0400; private const uint PROCESS_VM_READ = 0x0010; #endregion #region 拦截规则定义 /// /// 拦截规则类型 /// public enum InterceptType { /// /// 希沃白板3 桌面悬浮窗 /// SeewoWhiteboard3Floating, /// /// 希沃白板5 桌面悬浮窗 /// SeewoWhiteboard5Floating, /// /// 希沃白板5C 桌面悬浮窗 /// SeewoWhiteboard5CFloating, /// /// 希沃品课教师端 桌面悬浮窗 /// SeewoPincoSideBarFloating, /// /// 希沃品课教师端 画笔悬浮窗(包括PPT控件) /// SeewoPincoDrawingFloating, /// /// 希沃PPT小工具 /// SeewoPPTFloating, /// /// AiClass 桌面悬浮窗 /// AiClassFloating, /// /// 鸿合屏幕书写 /// HiteAnnotationFloating, /// /// 畅言智慧课堂 桌面悬浮窗 /// ChangYanFloating, /// /// 畅言智慧课堂 PPT悬浮窗 /// ChangYanPptFloating, /// /// 天喻教育云互动课堂 桌面悬浮窗(包括PPT控件) /// IntelligentClassFloating, /// /// 希沃桌面 画笔悬浮窗 /// SeewoDesktopAnnotationFloating, /// /// 希沃桌面 侧栏悬浮窗 /// SeewoDesktopSideBarFloating } /// /// 拦截规则 /// public class InterceptRule { public InterceptType Type { get; set; } public string ProcessName { get; set; } public string WindowTitlePattern { get; set; } public string ClassNamePattern { get; set; } public bool IsEnabled { get; set; } public bool RequiresAdmin { get; set; } public string Description { get; set; } } #endregion #region 私有字段 private readonly Dictionary _interceptRules; private readonly Dictionary _interceptedWindows; private readonly Timer _scanTimer; private readonly Dispatcher _dispatcher; private bool _isRunning; private bool _disposed; #endregion #region 公共属性 public bool IsRunning => _isRunning; #endregion #region 事件 public event EventHandler WindowIntercepted; public event EventHandler WindowRestored; #endregion #region 构造函数 public FloatingWindowInterceptor() { _interceptRules = new Dictionary(); _interceptedWindows = new Dictionary(); _dispatcher = Dispatcher.CurrentDispatcher; InitializeRules(); _scanTimer = new Timer(ScanForWindows, null, Timeout.Infinite, Timeout.Infinite); } #endregion #region 初始化 private void InitializeRules() { // 希沃白板3 桌面悬浮窗 _interceptRules[InterceptType.SeewoWhiteboard3Floating] = new InterceptRule { Type = InterceptType.SeewoWhiteboard3Floating, ProcessName = "EasiNote", WindowTitlePattern = "", ClassNamePattern = "", IsEnabled = true, RequiresAdmin = false, Description = "希沃白板3 桌面悬浮窗" }; // 希沃白板5 桌面悬浮窗 _interceptRules[InterceptType.SeewoWhiteboard5Floating] = new InterceptRule { Type = InterceptType.SeewoWhiteboard5Floating, ProcessName = "EasiNote", WindowTitlePattern = "", ClassNamePattern = "", IsEnabled = true, RequiresAdmin = false, Description = "希沃白板5 桌面悬浮窗" }; // 希沃白板5C 桌面悬浮窗 _interceptRules[InterceptType.SeewoWhiteboard5CFloating] = new InterceptRule { Type = InterceptType.SeewoWhiteboard5CFloating, ProcessName = "EasiNote5C", WindowTitlePattern = "", ClassNamePattern = "", IsEnabled = true, RequiresAdmin = false, Description = "希沃白板5C 桌面悬浮窗" }; // 希沃品课教师端 桌面悬浮窗 _interceptRules[InterceptType.SeewoPincoSideBarFloating] = new InterceptRule { Type = InterceptType.SeewoPincoSideBarFloating, ProcessName = "ClassIn", WindowTitlePattern = "", ClassNamePattern = "", IsEnabled = true, RequiresAdmin = false, Description = "希沃品课教师端 桌面悬浮窗" }; // 希沃品课教师端 画笔悬浮窗 _interceptRules[InterceptType.SeewoPincoDrawingFloating] = new InterceptRule { Type = InterceptType.SeewoPincoDrawingFloating, ProcessName = "ClassIn", WindowTitlePattern = "", ClassNamePattern = "", IsEnabled = true, RequiresAdmin = false, Description = "希沃品课教师端 画笔悬浮窗(包括PPT控件)" }; // 希沃PPT小工具 _interceptRules[InterceptType.SeewoPPTFloating] = new InterceptRule { Type = InterceptType.SeewoPPTFloating, ProcessName = "SeewoPPT", WindowTitlePattern = "", ClassNamePattern = "", IsEnabled = true, RequiresAdmin = false, Description = "希沃PPT小工具" }; // AiClass 桌面悬浮窗 _interceptRules[InterceptType.AiClassFloating] = new InterceptRule { Type = InterceptType.AiClassFloating, ProcessName = "ClassIn", WindowTitlePattern = "", ClassNamePattern = "", IsEnabled = true, RequiresAdmin = false, Description = "AiClass 桌面悬浮窗" }; // 鸿合屏幕书写 _interceptRules[InterceptType.HiteAnnotationFloating] = new InterceptRule { Type = InterceptType.HiteAnnotationFloating, ProcessName = "HiteVision", WindowTitlePattern = "", ClassNamePattern = "", IsEnabled = true, RequiresAdmin = false, Description = "鸿合屏幕书写" }; // 畅言智慧课堂 桌面悬浮窗 _interceptRules[InterceptType.ChangYanFloating] = new InterceptRule { Type = InterceptType.ChangYanFloating, ProcessName = "ClassIn", WindowTitlePattern = "", ClassNamePattern = "", IsEnabled = true, RequiresAdmin = true, Description = "畅言智慧课堂 桌面悬浮窗" }; // 畅言智慧课堂 PPT悬浮窗 _interceptRules[InterceptType.ChangYanPptFloating] = new InterceptRule { Type = InterceptType.ChangYanPptFloating, ProcessName = "ClassIn", WindowTitlePattern = "", ClassNamePattern = "", IsEnabled = true, RequiresAdmin = true, Description = "畅言智慧课堂 PPT悬浮窗" }; // 天喻教育云互动课堂 桌面悬浮窗 _interceptRules[InterceptType.IntelligentClassFloating] = new InterceptRule { Type = InterceptType.IntelligentClassFloating, ProcessName = "ClassIn", WindowTitlePattern = "", ClassNamePattern = "", IsEnabled = true, RequiresAdmin = false, Description = "天喻教育云互动课堂 桌面悬浮窗(包括PPT控件)" }; // 希沃桌面 画笔悬浮窗 _interceptRules[InterceptType.SeewoDesktopAnnotationFloating] = new InterceptRule { Type = InterceptType.SeewoDesktopAnnotationFloating, ProcessName = "Seewo", WindowTitlePattern = "", ClassNamePattern = "", IsEnabled = true, RequiresAdmin = false, Description = "希沃桌面 画笔悬浮窗" }; // 希沃桌面 侧栏悬浮窗 _interceptRules[InterceptType.SeewoDesktopSideBarFloating] = new InterceptRule { Type = InterceptType.SeewoDesktopSideBarFloating, ProcessName = "Seewo", WindowTitlePattern = "", ClassNamePattern = "", IsEnabled = true, RequiresAdmin = true, Description = "希沃桌面 侧栏悬浮窗" }; } #endregion #region 公共方法 /// /// 启动拦截器 /// public void Start(int scanIntervalMs = 5000) { if (_isRunning) return; _isRunning = true; _scanTimer.Change(0, scanIntervalMs); } /// /// 停止拦截器 /// public void Stop() { if (!_isRunning) return; _isRunning = false; _scanTimer.Change(Timeout.Infinite, Timeout.Infinite); // 恢复所有被拦截的窗口 RestoreAllWindows(); } /// /// 设置拦截规则 /// public void SetInterceptRule(InterceptType type, bool enabled) { if (_interceptRules.ContainsKey(type)) { _interceptRules[type].IsEnabled = enabled; } } /// /// 获取拦截规则 /// public InterceptRule GetInterceptRule(InterceptType type) { return _interceptRules.ContainsKey(type) ? _interceptRules[type] : null; } /// /// 获取所有拦截规则 /// public Dictionary GetAllRules() { return new Dictionary(_interceptRules); } /// /// 手动扫描一次 /// public void ScanOnce() { ScanForWindows(null); } /// /// 恢复所有被拦截的窗口 /// public void RestoreAllWindows() { var windowsToRestore = new List(_interceptedWindows.Keys); foreach (var hWnd in windowsToRestore) { RestoreWindow(hWnd); } } /// /// 恢复指定窗口 /// public bool RestoreWindow(IntPtr hWnd) { if (!_interceptedWindows.ContainsKey(hWnd)) return false; if (IsWindow(hWnd)) { ShowWindow(hWnd, SW_RESTORE); _interceptedWindows.Remove(hWnd); WindowRestored?.Invoke(this, new WindowRestoredEventArgs { WindowHandle = hWnd, InterceptType = _interceptedWindows[hWnd] }); return true; } _interceptedWindows.Remove(hWnd); return false; } #endregion #region 私有方法 private void ScanForWindows(object state) { if (!_isRunning) return; try { EnumWindows(EnumWindowsCallback, IntPtr.Zero); } catch (Exception ex) { // 记录错误但不中断扫描 LogHelper.WriteLogToFile($"扫描窗口时发生错误: {ex.Message}", LogHelper.LogType.Error); } } private bool EnumWindowsCallback(IntPtr hWnd, IntPtr lParam) { try { // 基本检查 if (!IsWindow(hWnd) || !IsWindowVisible(hWnd)) return true; // 检查是否已经被拦截 if (_interceptedWindows.ContainsKey(hWnd)) return true; // 获取窗口信息 var windowInfo = GetWindowInfo(hWnd); if (windowInfo == null) return true; // 检查窗口样式,过滤掉系统窗口和主窗口 var exStyle = GetWindowLong(hWnd, GWL_EXSTYLE); var style = GetWindowLong(hWnd, -16); // GWL_STYLE // 跳过工具窗口 if ((exStyle & WS_EX_TOOLWINDOW) != 0) { return true; } // 跳过主窗口(有标题栏和系统菜单的窗口) const uint WS_CAPTION = 0x00C00000; const uint WS_SYSMENU = 0x00080000; if ((style & WS_CAPTION) != 0 && (style & WS_SYSMENU) != 0) { return true; } // 检查窗口大小,跳过过大的窗口 var rect = new ForegroundWindowInfo.RECT(); GetWindowRect(hWnd, out rect); var width = rect.Right - rect.Left; var height = rect.Bottom - rect.Top; if (width > 600 || height > 400) { return true; } // 检查是否匹配拦截规则 foreach (var rule in _interceptRules.Values) { if (!rule.IsEnabled) continue; if (MatchesRule(windowInfo, rule)) { InterceptWindow(hWnd, rule); break; } } return true; } catch (Exception ex) { LogHelper.WriteLogToFile($"处理窗口时发生错误: {ex.Message}", LogHelper.LogType.Error); return true; } } private WindowInfo GetWindowInfo(IntPtr hWnd) { try { // 获取进程ID GetWindowThreadProcessId(hWnd, out uint processId); if (processId == 0) return null; // 获取进程信息 var process = Process.GetProcessById((int)processId); if (process == null) return null; // 获取窗口标题 var titleBuilder = new StringBuilder(256); GetWindowText(hWnd, titleBuilder, titleBuilder.Capacity); // 获取窗口类名 var classBuilder = new StringBuilder(256); GetClassName(hWnd, classBuilder, classBuilder.Capacity); return new WindowInfo { Handle = hWnd, ProcessId = processId, ProcessName = process.ProcessName, WindowTitle = titleBuilder.ToString(), ClassName = classBuilder.ToString(), Process = process }; } catch { return null; } } private bool MatchesRule(WindowInfo windowInfo, InterceptRule rule) { try { // 检查进程名(如果指定了进程名) if (!string.IsNullOrEmpty(rule.ProcessName)) { if (!windowInfo.ProcessName.ToLower().Contains(rule.ProcessName.ToLower())) { return false; } } // 检查窗口标题(如果指定了模式) if (!string.IsNullOrEmpty(rule.WindowTitlePattern)) { if (!windowInfo.WindowTitle.ToLower().Contains(rule.WindowTitlePattern.ToLower())) { return false; } } // 检查类名(如果指定了模式) if (!string.IsNullOrEmpty(rule.ClassNamePattern)) { if (!windowInfo.ClassName.ToLower().Contains(rule.ClassNamePattern.ToLower())) { return false; } } // 如果所有检查都通过,就认为是目标窗口 return true; } catch (Exception ex) { LogHelper.WriteLogToFile($"匹配规则时发生错误: {ex.Message}", LogHelper.LogType.Error); return false; } } private void InterceptWindow(IntPtr hWnd, InterceptRule rule) { try { // 使用多种方法隐藏窗口 // 方法1:移动到屏幕外 SetWindowPos(hWnd, IntPtr.Zero, -2000, -2000, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_HIDEWINDOW); // 方法2:最小化窗口 ShowWindow(hWnd, SW_MINIMIZE); // 方法3:隐藏窗口 ShowWindow(hWnd, SW_HIDE); // 记录拦截的窗口 _interceptedWindows[hWnd] = rule.Type; // 触发事件 WindowIntercepted?.Invoke(this, new WindowInterceptedEventArgs { WindowHandle = hWnd, InterceptType = rule.Type, Rule = rule, WindowTitle = GetWindowTitle(hWnd) }); } catch (Exception ex) { LogHelper.WriteLogToFile($"拦截窗口时发生错误: {ex.Message}", LogHelper.LogType.Error); } } private string GetWindowTitle(IntPtr hWnd) { try { var titleBuilder = new StringBuilder(256); GetWindowText(hWnd, titleBuilder, titleBuilder.Capacity); return titleBuilder.ToString(); } catch { return "Unknown"; } } private bool IsMainWindow(IntPtr hWnd) { try { // 检查是否有父窗口 var parent = GetWindow(hWnd, 4); // GW_OWNER if (parent != IntPtr.Zero) return false; // 检查窗口样式 var style = GetWindowLong(hWnd, -16); // GWL_STYLE var exStyle = GetWindowLong(hWnd, GWL_EXSTYLE); // 主窗口通常有 WS_CAPTION 和 WS_SYSMENU const uint WS_CAPTION = 0x00C00000; const uint WS_SYSMENU = 0x00080000; if ((style & WS_CAPTION) != 0 && (style & WS_SYSMENU) != 0) { return true; // 这可能是主窗口 } // 检查窗口大小,主窗口通常比较大 var rect = new ForegroundWindowInfo.RECT(); GetWindowRect(hWnd, out rect); var width = rect.Right - rect.Left; var height = rect.Bottom - rect.Top; // 如果窗口很大,可能是主窗口 if (width > 800 && height > 600) { return true; } return false; } catch { return false; } } #endregion #region 辅助类 private class WindowInfo { public IntPtr Handle { get; set; } public uint ProcessId { get; set; } public string ProcessName { get; set; } public string WindowTitle { get; set; } public string ClassName { get; set; } public Process Process { get; set; } } #endregion #region 事件参数类 public class WindowInterceptedEventArgs : EventArgs { public IntPtr WindowHandle { get; set; } public InterceptType InterceptType { get; set; } public InterceptRule Rule { get; set; } public string WindowTitle { get; set; } } public class WindowRestoredEventArgs : EventArgs { public IntPtr WindowHandle { get; set; } public InterceptType InterceptType { get; set; } } #endregion #region IDisposable public void Dispose() { if (_disposed) return; Stop(); _scanTimer?.Dispose(); // 恢复所有被拦截的窗口 RestoreAllWindows(); _disposed = true; } #endregion } }