WPF 调用 Win32的SetWindowDisplayAffinity 函数 实现捕获屏幕时,过滤指定的窗口

一、背景

最近在开发互动传屏的需求,使用了WGC捕获桌面屏幕的方式,实现屏幕的捕获和录制。在验收阶段,产品的验收的建议是:需要过滤自身的窗口,跟腾讯会议一样,进行互动传屏。

1、腾讯会议共享屏幕的窗口如下图

腾讯会议的发起者的主窗口在其他设备端,是看不到的,直接透明下去。还有会议的录屏的全屏四个边角的绿框也是看不到的

2、我们自己开发的屏幕捕获的效果如下图:

窗口在捕获之后,都是层层传递下去,类似直播的效果一样。

二、介绍SetWindowDisplayAffinity 函数

其实微软的Win32 API 有函数 实现:SetWindowDisplayAffinity 函数 (winuser.h) - Win32 apps | Microsoft Learn,支持 Windows 7 中新增的窗口内容保护功能。

此功能使应用程序能够保护自己的屏幕窗口内容,使其免受通过一组特定的公共操作系统功能和 API 捕获或复制

SetWindowDisplayAffinity 函数 (winuser.h)

指定窗口内容的显示位置。

语法

C++

复制代码
BOOL SetWindowDisplayAffinity(
  [in] HWND  hWnd,
  [in] DWORD dwAffinity
);

参数

[in] hWnd

类型:HWND

顶级窗口的句柄。 窗口必须属于当前进程。

[in] dwAffinity

类型:DWORD

指定窗口内容的显示位置的显示相关性设置。

此参数的取值可为下列值之一:

含义
WDA_NONE 0x00000000 对窗口的显示位置没有限制。
WDA_MONITOR 0x00000001 窗口内容仅显示在监视器上。 在其他任何位置,窗口都会显示,其中不显示任何内容。
WDA_EXCLUDEFROMCAPTURE 0x00000011 窗口仅显示在监视器上。 在其他任何位置,窗口根本不显示。 此相关性的一个用途是用于显示视频录制控件的窗口,以便控件不包括在捕获中。 Windows 10版本 2004 中引入。 请参阅有关以前版本的 Windows 兼容性的备注。

返回值

类型: BOOL

如果函数成功,则返回 TRUE;例如,在非顶级窗口上进行函数调用时,它将返回 FALSE 。 要获得更多的错误信息,请调用 GetLastError。

注解

此函数和 GetWindowDisplayAffinity 旨在支持 Windows 7 中新增的窗口内容保护功能。 此功能使应用程序能够保护自己的屏幕窗口内容,使其免受通过一组特定的公共操作系统功能和 API 捕获或复制。 但是,仅当桌面窗口管理器 (DWM) 组合桌面时,它才有效。

请务必注意,与安全功能或数字版权管理 (DRM) 实现不同,无法保证使用 SetWindowDisplayAffinity 和 GetWindowDisplayAffinity 以及其他必要功能(如 DwmIsCompositionEnabled)将严格保护窗口内容,例如有人拍摄屏幕照片。

从 Windows 10 版本 2004 开始,WDA_EXCLUDEFROMCAPTURE 是受支持的值。 在以前版本的 Windows 上将显示相关性设置为WDA_EXCLUDEFROMCAPTURE的行为就像应用了WDA_MONITOR一样。

要求

要求
最低受支持的客户端 Windows 7 [仅限桌面应用]
最低受支持的服务器 Windows Server 2008 R2 [仅限桌面应用]
目标平台 Windows
标头 winuser.h (包括 Windows.h)
Library User32.lib
DLL User32.dll
API 集 在 Windows 8.1) 中引入的 ext-ms-win-ntuser-window-l1-1-1 (

另请参阅

SetWindowDisplayAffinityWindows

三、使用SetWindowDisplayAffinity 函数

1、定义 SetWindowDisplayAffinity 函数

复制代码
 private const uint WDA_NONE = 0x00000000;
 private const uint WDA_MONITOR = 0x00000001;
 private const uint WDA_EXCLUDEFROMCAPTURE = 0x00000011;

 [DllImport("user32.dll")]
 public static extern uint SetWindowDisplayAffinity(IntPtr hWnd, uint dwAffinity);

2、在MainWindows 的 SourceInitialized 事件调用(只要能获取得到窗口的句柄即可),由于我的需求是在捕获屏幕的时候,不需要显示,所以我这边调用的是:WDA_EXCLUDEFROMCAPTURE(0x00000011)

复制代码
 public MainWindow()
 {
     InitializeComponent();

     SourceInitialized += MainWindow_SourceInitialized;
 }

 private IntPtr currentWindowHandle;

 private void MainWindow_SourceInitialized(object sender, EventArgs e)
 {
     currentWindowHandle = new WindowInteropHelper(this).Handle;
     var result =  SetWindowDisplayAffinity(currentWindowHandle, WDA_EXCLUDEFROMCAPTURE);
 }

 private const uint WDA_NONE = 0x00000000;
 private const uint WDA_MONITOR = 0x00000001;
 private const uint WDA_EXCLUDEFROMCAPTURE = 0x00000011;

 [DllImport("user32.dll")]
 public static extern uint SetWindowDisplayAffinity(IntPtr hWnd, uint dwAffinity);

3、调用的效果如下:

捕获的窗口已经没有显示主窗口了

参考资料:

1、SetWindowDisplayAffinity 函数 (winuser.h) - Win32 apps | Microsoft Learn

相关推荐
TracyCoder1232 小时前
RocketMQ技术原理简单解析:从架构到核心流程
架构·wpf·rocketmq
烽火聊员4 小时前
SSLSocket 服务器端WPF C#测试代码
开发语言·c#·wpf·ssl
暮雪倾风5 小时前
【WPF开发】加载solidworks的3D模型
wpf
Macbethad5 小时前
高性能 CANopen 主站程序技术方案 (基于 WPF)
网络协议·wpf·信息与通信
Macbethad15 小时前
使用WPF编写一个工控软件设置界面
wpf
wuli_滔滔21 小时前
【探索实战】深入浅出:使用Kurator Fleet实现跨云集群的统一应用分发
架构·wpf·kurator·fleet
松☆1 天前
Flutter 与 OpenHarmony 深度集成:自定义 MethodChannel 插件开发全指南
flutter·wpf
Aevget1 天前
界面控件DevExpress WPF中文教程:Data Grid - 虚拟源限制
hadoop·wpf·界面控件·devexpress·ui开发
听风吟丶1 天前
Java 微服务日志实战:ELK+SkyWalking 构建全链路日志监控与智能告警体系某电商平台曾因日志问题陷入
wpf