为.NET应用添加截图功能

简介: 本文介绍了 .NET 实现截图功能的思路和过程,如果你仅想了解最后的解决方案,可以直接查看文章末尾。

截图的功能我们应该都经常使用,在开发软件时,我们有时也或多或少需要提供这方面的功能,无论是为用户更方便提供远程诊断,还是获取用户的选择区域,亦或是提供某些功能上的辅助。

开发截图无非就这几种选择:教用户使用截图工具、自行开发一个、使用第三方库。

教用户使用

教的成本无疑是最低的,但是不知道用户那边会发生什么,存在很大的不确定性。截图软件除了我们经常用的聊天工具和系统自带的 Win + Ctrl + S外,我用起来感觉最好的还是 C++ 写的开源软件 flameshot ,功能非常强大。

flameshot

使用的第三方的截图软件,不仅有教的成本,还会打断用户对本身软件的一个使用体验。教用户使用最好还是用系统自带的 Win + Ctrl + S截图,已经可以满足基本的截图需求。

自行开发

自行开发的原理也非常简单:创建一个半透明的全屏无边框窗体,记录鼠标在窗体上的框选矩形位置,使用CopyFromScreen获取该位置的屏幕图片即可。

以上只是针对单个显示器的情况,若有多个显示器,则需要增加鼠标所在显示器的逻辑。

虽然听起来不难,但代码实现起来还是有许多要注意的细节。简单的矩形截图实现不难,难得是让用户易用,易接受,毕竟聊天软件已经帮你培养了用户习惯。

使用第三方库

CSkin 是我在 2012 年就在使用的一款界面库,在 WinForm 无疑是软件 UI 美化的王者,可以直接作出和 PC 端 QQ 一样的界面体验。库里也提供了截图工具 FrmCapture,没中不足的是,在多显示器场景下会报错,无法正常使用,代码库也有 4 年没有更新了。

private FrmCapture m_frmCapture;
if (m_frmCapture == null || m_frmCapture.IsDisposed)
{
    m_frmCapture = new FrmCapture();
}
m_frmCapture.IsCaptureCursor = false;
// 截图结束事件
m_frmCapture.Disposed += M_frmCapture_Disposed;
m_frmCapture.Show();

HandyControl 和在 nuget 上搜索到的 ScreenCapturerSharp 虽然也可以实现截图功能,但都无法处理多显示器的场景。HandyControl 社区活跃,其使用体验会比较好。ScreenCapturerSharp 提供了类似 QQ 的截图工具库,在 UI 上稍差一些。

如何又快又好又容易

如果只是获取截图,有没有更简单的方式呢?我们只需要模拟按键 Win + Ctrl + S 就可以了呀,然后通过剪贴板获取到截图。说起来容易,但是事情其实并没有那么简单。

首先 SendKeys 就不支持发送 Windows 徽标按键,我们需要通过 WinAPI keybd_event 来替代实现,然后还要获取到截图结束的事件。

[DllImport("user32.dll", SetLastError = true)]
static extern void keybd_event(byte bVk, byte bScan, uint dwFlags, UIntPtr dwExtraInfo);

其实上面是一个保底的通用方案,我们可以自行启动截图软件,启动截图软件读取剪贴板Clipboard.GetImage() 一套结束,无缝无感,堪称完美:

Process snippingToolProcess = new Process()
{
    StartInfo = new ProcessStartInfo("C:\\Windows\\system32\\SnippingTool.exe", "/clip"),
    EnableRaisingEvents = true,
};
snippingToolProcess.Exited += SnippingToolProcess_Exited;
snippingToolProcess.Start();

事情其实远没有那么简单,直到我在 Win11 用了我的软件。才意识到,这只是可以在 Win10 的 64 位操作系统使用。SnippingTool /clip 这样带参数启动在 Win11 不支持了,这个路径下的 exe 还被删除了。

虽然你可以在 Win11 通过控制台使用SnippingTool /clip启动截图软件,但是并不会直接进入截图流程,而是打开软件主界面。

仔细研究你会发现,Win11 的截图其实已经是 UWP 应用了,就算你吧 Win10 的 SnippingTool.exe 复制到 Win11 也是报错,无法使用的,所以你也不可能在自己的软件打包带上它。

经过几番折腾,我在微软社区提问和提交反馈( Win + F 的时候我觉得这个软件是不是这样启动直接就先截了个屏 ),但是没有找到新版本截图的启动参数。最后直到我前几天发现 Microsoft Learn 的文章 启动屏幕截取 - UWP applications。在 UWP 里使用这么简单嘛,使用 LaunchUriAsync 就可以了。

bool result = await Windows.System.Launcher.LaunchUriAsync(new Uri("ms-screenclip:"));

有了 URI 的方式,一切就变简单了,你甚至可以在浏览器里调用截图,放一个超链接,或者直接在浏览器地址栏粘贴ms-screenclip:后回车打开截图。

之后我们只需要监听进程结束就可以了,这里需要说明的是,不是启动的进程,而是截图的进程,下面直接上在 WinForm 中使用的代码:

var psi = new ProcessStartInfo()
{
    UseShellExecute = true,
    FileName = "ms-screenclip:"
};
Process.Start(psi);

// 获取 ScreenClippingHost 这个截图进程的结束事件
var snippingToolProcess = Process.GetProcessesByName("ScreenClippingHost")[0];
snippingToolProcess.EnableRaisingEvents = true;
snippingToolProcess.Exited += SnippingToolProcess_Exited;

SnippingToolProcess_Exited 事件:

private void SnippingToolProcess_Exited(object? sender, EventArgs e)
{
    this.BeginInvoke(new Action(() =>
    {
        var img = Clipboard.GetImage();
    }));
}
相关文章
|
2月前
|
存储 Shell Linux
快速上手基于 BaGet 的脚本自动化构建 .net 应用打包
本文介绍了如何使用脚本自动化构建 `.net` 应用的 `nuget` 包并推送到指定服务仓库。首先概述了 `BaGet`——一个开源、轻量级且高性能的 `NuGet` 服务器,支持多种存储后端及配置选项。接着详细描述了 `BaGet` 的安装、配置及使用方法,并提供了 `PowerShell` 和 `Bash` 脚本实例,用于自动化推送 `.nupkg` 文件。最后总结了 `BaGet` 的优势及其在实际部署中的便捷性。
123 10
|
4天前
|
消息中间件 监控 数据可视化
基于.NET开源、功能强大且灵活的工作流引擎框架
基于.NET开源、功能强大且灵活的工作流引擎框架
|
7天前
|
开发框架 监控 .NET
【Azure App Service】部署在App Service上的.NET应用内存消耗不能超过2GB的情况分析
x64 dotnet runtime is not installed on the app service by default. Since we had the app service running in x64, it was proxying the request to a 32 bit dotnet process which was throwing an OutOfMemoryException with requests >100MB. It worked on the IaaS servers because we had the x64 runtime install
|
20天前
|
JSON 算法 安全
JWT Bearer 认证在 .NET Core 中的应用
【10月更文挑战第30天】JWT(JSON Web Token)是一种开放标准,用于在各方之间安全传输信息。它由头部、载荷和签名三部分组成,用于在用户和服务器之间传递声明。JWT Bearer 认证是一种基于令牌的认证方式,客户端在请求头中包含 JWT 令牌,服务器验证令牌的有效性后授权用户访问资源。在 .NET Core 中,通过安装 `Microsoft.AspNetCore.Authentication.JwtBearer` 包并配置认证服务,可以实现 JWT Bearer 认证。具体步骤包括安装 NuGet 包、配置认证服务、启用认证中间件、生成 JWT 令牌以及在控制器中使用认证信息
|
3天前
|
XML 开发框架 .NET
.NET 9 中 LINQ 新增功能实操
.NET 9 中 LINQ 新增功能实操
|
4天前
|
网络协议 Unix Linux
精选2款C#/.NET开源且功能强大的网络通信框架
精选2款C#/.NET开源且功能强大的网络通信框架
|
4天前
|
开发框架 JavaScript 前端开发
2024年全面且功能强大的.NET快速开发框架推荐,效率提升利器!
2024年全面且功能强大的.NET快速开发框架推荐,效率提升利器!
|
4天前
|
网络协议 网络安全 Apache
一个整合性、功能丰富的.NET网络通信框架
一个整合性、功能丰富的.NET网络通信框架
|
7天前
|
消息中间件 开发框架 .NET
.NET 8 强大功能 IHostedService 与 BackgroundService 实战
【11月更文挑战第7天】本文介绍了 ASP.NET Core 中的 `IHostedService` 和 `BackgroundService` 接口及其用途。`IHostedService` 定义了 `StartAsync` 和 `StopAsync` 方法,用于在应用启动和停止时执行异步操作,适用于资源初始化和清理等任务。`BackgroundService` 是 `IHostedService` 的抽象实现,简化了后台任务的编写,通过 `ExecuteAsync` 方法实现长时间运行的任务逻辑。文章还提供了创建和注册这两个服务的实战步骤,帮助开发者在实际项目中应用这些功能。
|
21天前
.NET 4.0下实现.NET4.5的Task类相似功能组件
【10月更文挑战第29天】在.NET 4.0 环境下,可以使用 `BackgroundWorker` 类来实现类似于 .NET 4.5 中 `Task` 类的功能。`BackgroundWorker` 允许在后台执行耗时操作,同时不会阻塞用户界面线程,并支持进度报告和取消操作。尽管它有一些局限性,如复杂的事件处理模型和不灵活的任务管理方式,但在某些情况下仍能有效替代 `Task` 类。