c# wpf 利用截屏键实现截屏功能

简介: 原文:c# wpf 利用截屏键实现截屏功能     最近做一个wpf程序需要截图功能,查找资料费了一些曲折,跟大家分享一下。
原文: c# wpf 利用截屏键实现截屏功能

    最近做一个wpf程序需要截图功能,查找资料费了一些曲折,跟大家分享一下。

    先是找到了这样一份代码:

    static class ScreenCut

    {

        public static System.Drawing.Bitmap GetScreenSnapshot()

        {

            System.Drawing.Rectangle rc = System.Windows.Forms.SystemInformation.VirtualScreen;

            System.Drawing.Bitmap bitmap = new System.Drawing.Bitmap(rc.Width, rc.Height, System.Drawing.Imaging.PixelFormat.Format32bppArgb);

            using (System.Drawing.Graphics g = System.Drawing.Graphics.FromImage(bitmap))

            {

                g.CopyFromScreen(rc.X, rc.Y, 0, 0, rc.Size, System.Drawing.CopyPixelOperation.SourceCopy);

            }

            return bitmap;

        }

        public static BitmapSource ToBitmapSource(this System.Drawing.Bitmap bmp)

        {

            BitmapSource returnSource;

            try

            {

                returnSource = System.Windows.Interop.Imaging.CreateBitmapSourceFromHBitmap(bmp.GetHbitmap(), IntPtr.Zero, Int32Rect.Empty, BitmapSizeOptions.FromEmptyOptions());

            }

            catch

            {

                returnSource = null;

            }

            return returnSource;

        }

 

        public static BitmapSource CopyFromScreenSnapshot(BitmapSource screenSnapshot, Rect region)

        {

            var sourceRect = new System.Drawing.Rectangle((int)region.Left, (int)region.Top, (int)region.Width, (int)region.Height);

            var destRect = new System.Drawing.Rectangle(0, 0, sourceRect.Width, sourceRect.Height);

 

            if (screenSnapshot != null)

            {

                var bitmap = new System.Drawing.Bitmap(sourceRect.Width, sourceRect.Height, System.Drawing.Imaging.PixelFormat.Format32bppArgb);

                using (System.Drawing.Graphics g = System.Drawing.Graphics.FromImage(bitmap))

                {

                    g.DrawImage(bitmap, destRect, sourceRect.Left, sourceRect.Top, sourceRect.Width, sourceRect.Height, System.Drawing.GraphicsUnit.Pixel);

                }

 

                return bitmap.ToBitmapSource();

            }

 

            return null;

        }

}

 

调用时包装一下,下面的函数截下屏幕指定区域并保存:

    System.Drawing.Image takePicture(Rect wnd, String saveLocation, String pictureName)

{    

        BitmapSource bitmapScr = ScreenCut.CopyFromScreenSnapshot(ScreenCut.ToBitmapSource(ScreenCut.GetScreenSnapshot()),wnd);

        PngBitmapEncoder encoder = new PngBitmapEncoder();

        encoder.Frames.Add(BitmapFrame.Create(bitmapScr));

        FileStream fileStream = new FileStream(saveLocation + "\\" + pictureName ".png", FileMode.Create, FileAccess.Write);

        encoder.Save(fileStream);

        fileStream.Close();

}

 

但实际效果保存下来的图是空的,经验证前两个函数没有问题。于是我查资料找到了另一种切割BitmapSource的方法,将第三个函数改成这样:

BitmapSource CopyFromScreenSnapshot(BitmapSource screenSnapshot, Rect region)

{

        Int32Rect window = new Int32Rect(region.X, region.Y, region.Width, region.Height);

        BitmapSource bitmapScr = ScreenCut.ToBitmapSource(ScreenCut.GetScreenSnapshot());

        //计算Stride

        int stride = bitmapScr.Format.BitsPerPixel * window.Width / 8;

        //声明字节数组

        byte[] data = new byte[window.Height * stride];

        //调用CopyPixels

        bitmapScr.CopyPixels(window, data, stride, 0);

        PngBitmapEncoder encoder = new PngBitmapEncoder();

    return BitmapSource.Create(window.Width, window.Height, 0, 0, PixelFormats.Bgr32, null, data, stride);

}

 

这样总算能截到屏了。不过后来发现在某些主题下有些程序的窗口在截下的图中是透明的。这个方法还是不够完美。

我想到了利用键盘的截屏键截图再进行剪切的方法。分为三步:1、模拟按下截屏键 2、从剪贴板获取图片 3、截取和保存图片。

对于第一点,利用SendMessage函数发送WM_KEY事件没有用,不过我找到了另一个模拟键盘输入的方法:

using System.Runtime.InteropServices;

    [DllImport("user32.dll")]

    static extern void keybd_event

    (

        byte bVk,// 虚拟键值

        byte bScan,// 硬件扫描码    

        uint dwFlags,// 动作标识    

        IntPtr dwExtraInfo// 与键盘动作关联的辅加信息    

);

 

上面这个函数对本程序发送按钮事件,模拟截屏键的话像下面这样写,wpf需要在项目添加引用System.Windows.Forms

const int VK_SNAPSHOT = 0x2C;

    public void PrintScreen()

    {

        keybd_event((byte)VK_SNAPSHOT, 0, 0x0, IntPtr.Zero);//down  

        System.Windows.Forms.Application.DoEvents();//强制窗口响应按钮事件

        keybd_event((byte)VK_SNAPSHOT, 0, 0x2, IntPtr.Zero);//up  

        System.Windows.Forms.Application.DoEvents();

}

 

接下来从剪贴板获取图片,wpf需要在项目添加引用System.Windows.Forms  System.Drawing

    if (System.Windows.Forms.Clipboard.ContainsImage())

    {

       System.Drawing.Image image = System.Windows.Forms.Clipboard.GetImage();

}

 

然后剪切。

    System.Drawing.Image cutPicture(System.Drawing.Image image, Int32Rect window)

    {

        PrintScreen();

        if (window.X + window.Width > image.Width)

            window.Width = image.Width - window.X;

        if (window.Y + window.Height > image.Height)

            window.Height = image.Height - window.Y;

        if (image == null)

        {

            //新建一个bmp图片

            System.Drawing.Image bitmap = new System.Drawing.Bitmap(window.Width, window.Height);

            //新建一个画板

            System.Drawing.Graphics graphic = System.Drawing.Graphics.FromImage(bitmap);

            //设置高质量查值法

            graphic.InterpolationMode = System.Drawing.Drawing2D.InterpolationMode.High;

            //设置高质量,低速度呈现平滑程度

            graphic.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.HighQuality;

            //清空画布并以透明背景色填充

            graphic.Clear(System.Drawing.Color.Transparent);

            //在指定位置并且按指定大小绘制原图片的指定部分

            graphic.DrawImage(image, new System.Drawing.Rectangle(0, 0, window.Width, window.Height), new System.Drawing.Rectangle(window.X, window.Y, window.Width, window.Height), System.Drawing.GraphicsUnit.Pixel);

            return bitmap;

        }

        else

        {

            return null;

        }

}

 

最后包装一下

    System.Drawing.Image cutScreen(Int32Rect window)

    {

        PrintScreen();

        Thread.Sleep(1000);

        if (System.Windows.Forms.Clipboard.ContainsImage())

        {

            System.Drawing.Image image = cutPicture(System.Windows.Forms.Clipboard.GetImage(), Int32Rect window);

            return image;

        }

        else

        {

            Console.WriteLine("clipboard doesn't contain picture");

            return null;

        }

}

System.Drawing.Image类有实例函数Save,保存很简单,就不细说了。

资料来源:

http://www.cnblogs.com/zhouyinhui/archive/2010/08/20/1804762.html

https://www.mgenware.com/blog/?p=285

http://blog.csdn.net/testcs_dn/article/details/39762017

目录
相关文章
|
3月前
|
网络协议 网络安全 C#
基于 WPF 开发的简约,功能强大的终端模拟器
基于 WPF 开发的简约,功能强大的终端模拟器 前言今天大姚给大家推荐一款基于 WPF 开发的简约,功能强大的终端模拟器:ModengTerm。项目介绍ModengTerm是一款基于 WPF 开发的简约,功能强大的终端模拟器,可以用来连接SSH服务器,串口,TCP服务器,Windows命令行等。项目功能支持与SSH服务器,串口,Windows命令行进行交互。可以保存会话信息,方便下次直接登录。支持将终端内容导出为txt和html格式。根据关键字/正则表达式进行历史记录的查找。同步输入功能、历史记录、度可定制化的颜色主题、实时记录日志功能等。项目源码运行设置ModengTerm为启动项目运行:
|
4月前
|
前端开发 C# Windows
在WPF程序中实现PropertyGrid功能
【11月更文挑战第15天】PropertyGrid 是一个用户界面组件,用于直观地查看和编辑对象属性。在 WPF 中可通过组合 Expander 和 DataGrid 实现基本功能,或使用第三方库 PropertyTools 获得更强大特性,包括属性验证和类型特定编辑器。
214 3
|
4月前
|
开发框架 C# iOS开发
基于C#开源、功能强大、灵活的跨平台开发框架 - Uno Platform
基于C#开源、功能强大、灵活的跨平台开发框架 - Uno Platform
140 3
|
4月前
|
网络协议 Unix Linux
精选2款C#/.NET开源且功能强大的网络通信框架
精选2款C#/.NET开源且功能强大的网络通信框架
131 0
|
4月前
|
传感器 IDE 测试技术
C#一分钟浅谈:Visual Studio IDE 高级功能
【10月更文挑战第24天】本文从初学者角度介绍了 Visual Studio 的高级功能,包括安装与配置、创建项目、运行与调试、常见问题及解决方案(如代码格式化、重构、导航、单元测试、代码分析),以及智能感知、Live Unit Testing、代码生成和代码片段等高级功能,帮助开发者提高效率和代码质量。
85 1
|
5月前
|
开发框架 Cloud Native .NET
10 个 C# 关键字和功能
10 个 C# 关键字和功能
84 8
|
5月前
|
物联网 C#
【C#】简单的蓝牙通讯功能实现
【C#】简单的蓝牙通讯功能实现
146 0
|
5月前
|
C#
如何使用c# 实现断点续传功能
如何使用c# 实现断点续传功能
51 0
|
5月前
|
网络协议 C#
C#:简化版的实现断点续传功能
C#:简化版的实现断点续传功能
69 0
|
5月前
|
监控 前端开发 安全
C#一分钟浅谈:文件上传与下载功能实现
【10月更文挑战第2天】在Web应用开发中,文件的上传与下载是常见需求。本文从基础入手,详细讲解如何在C#环境下实现文件上传与下载。首先介绍前端表单设计及后端接收保存方法,使用`<input type="file">`与`IFormFile`接口;接着探讨错误处理与优化策略,如安全性验证和路径管理;最后讲解文件下载的基本步骤,包括确定文件位置、设置响应头及发送文件流。此外,还提供了进阶技巧,如并发处理、大文件分块上传及进度监控,帮助开发者构建更健壮的应用系统。
263 15