C#实现截图工具小项目

 更新时间:2021年07月26日 10:45:14   作者:*八步赶蝉*  
这篇文章主要为大家详细介绍了C#实现截图工具小项目,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了C#实现截图工具小项目的具体代码,供大家参考,具体内容如下

1.起因

一直用的截图是qq的截图,所以想要实现一个简单点的截图,为了方便。

2.思路

讲一下实现流程。

1、主窗体,上有截图按钮,点击进入截图窗体
2、在截图窗体中,背景设置为全屏幕的截图图片,无边框,窗体最大化,这时你看到的就是一张屏幕图,其实是一个窗体,然后我们将在这个窗体中截取图片,其实主要就是画板Graphics的使用,截取完之后图片将保存到剪切板。

3.代码

热键注册类   HotKey.cs

using System;
using System.Runtime.InteropServices;
using System.Windows.Forms;
 
namespace test {
    /// <summary>
    /// 热键类
    /// </summary>
    public class HotKey {
        /// <summary>
        /// 如果函数执行成功,返回值不为0,如果执行失败,返回值为0
        /// </summary>
        /// <returns></returns>
        [DllImport("user32.dll", SetLastError = true)]
        public static extern bool RegisterHotKey(
            IntPtr hWnd,           // 窗口的句柄, 当热键按下时,会产生WM_HOTKEY信息,该信息该会发送该窗口句柄
            int id,                      // 定义热键ID,属于唯一标识热键的作用
            uint fsModifiers,            // 热键只有在按下Alt、 Ctrl、Shift、Windows等键时才会生效,即才会产生WM_HOTKEY信息
            Keys vk                   // 虚拟键,即按了Alt+Ctrl+ X ,X就是代表虚拟键
            );
 
        [DllImport("user32.dll", SetLastError = true)]
        public static extern bool UnregisterHotKey(
            IntPtr hWnd,                   // 窗口句柄
            int id                         // 要取消热键的ID
            );
    }
}

主窗体  Form1.cs

using System;
using System.Drawing;
using System.Drawing.Imaging;
using System.Threading;
using System.Timers;
using System.Windows.Forms;
 
namespace test {
    public partial class Form1 : Form {
 
        [Flags]
        public enum KeyModifiers { //定义热键值字符串(热键值是系统规定的,不能改变)
            None = 0,
            Alt = 1,
            Ctrl = 2,
            Shift = 4,
            WindowsKey = 8
        }
 
        public Form1() {
            InitializeComponent();
        }
 
        //窗体加载时-注册快捷键
        private void Form1_Load(object sender, EventArgs e) {
            uint ctrlHotKey = (uint)(KeyModifiers.Alt | KeyModifiers.Ctrl);
            // 注册热键为Alt+Ctrl+A, "100"为唯一标识热键
            HotKey.RegisterHotKey(Handle,100,ctrlHotKey,Keys.A);
        }
 
        //截图按钮
        private void button1_Click(object sender, EventArgs e) {
            if (this.WindowState != FormWindowState.Minimized) {
                this.WindowState = FormWindowState.Minimized;
                Thread.Sleep(200);
            }
            int swidth = Screen.PrimaryScreen.Bounds.Width;
            int sheight = Screen.PrimaryScreen.Bounds.Height;
            Bitmap btm = new Bitmap(swidth,sheight); //空图与屏幕同大小
            Graphics g = Graphics.FromImage(btm); //空图的画板
            g.CopyFromScreen(new Point(0,0),new Point(0,0),new Size(swidth,sheight)); //将屏幕内容复制到空图
            Cutter cutter = new Cutter(btm); //传送截图
            cutter.FormBorderStyle = FormBorderStyle.None; //截图全屏,无边框
            cutter.BackgroundImage = btm; //新的窗体截图做背景
            cutter.Show();
        }
        private void tiaoZ(object sender, ElapsedEventArgs e) {
        }
        
 
        //窗体关闭-取消热键
        private void Form1_FormClosing(object sender, FormClosingEventArgs e) {
            HotKey.UnregisterHotKey(Handle,100);
        }
 
        //快捷键按下执行的事件
        private void GlobalKeyProcess() {
            this.WindowState = FormWindowState.Minimized;
            Thread.Sleep(200);
            button1.PerformClick();
        }
 
        //重写。监视系统消息,调用对应方法
        protected override void WndProc(ref Message m) {
            const int WM_HOTKEY = 0x0312;
            //如果m.Msg的值为0x0312(我也不知道为什么是0x0312)那么表示用户按下了热键
            switch (m.Msg) {
                case WM_HOTKEY:
                    if (m.WParam.ToString().Equals("100")) {
                        GlobalKeyProcess();
                    }
                    //todo  其它热键
                    break;
            }
            // 将系统消息传递自父类的WndProc
            base.WndProc(ref m);
        }
    }
}

截图窗体-核心 Cutter.cs

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
 
namespace test {
    public partial class Cutter : Form {
        Bitmap screenBtmp = null; //电脑屏幕的截图
 
        public Cutter(Bitmap btm) {
            InitializeComponent();
            screenBtmp = btm;
        }
 
        //鼠标右键退出
        private void Cutter_MouseClick(object sender, MouseEventArgs e) {
            if (e.Button == MouseButtons.Right) {
                this.DialogResult = DialogResult.OK;
                this.Close();
            }
        }
 
        bool CatchStart = false; //自由截图开始
        Point downPoint; //初始点
        //鼠标左键按下-开始自由截图
        private void Cutter_MouseDown(object sender, MouseEventArgs e) {
            if (e.Button == MouseButtons.Left) {
                if (!CatchStart) {
                    CatchStart = true;
                    downPoint = new Point(e.X,e.Y); //初始点
                }
            }
        }
 
        Rectangle catchRec;//存放截取范围
        //鼠标移动-绘制自由截图路径
        private void Cutter_MouseMove(object sender, MouseEventArgs e) { //路径绘制,核心
            if (CatchStart) {
                //
                //二次缓冲
                //不是直接在控件的背景画板上进行绘制鼠标移动路径,那样会造成绘制很多路径,因为前面绘制的路径还在
                //而是在内存中每移动一次鼠标就创建一张和屏幕截图一样的新BImtap,在这个Bitmap中绘制鼠标移动路径
                //然后在窗体背景画板上,绘制这个新的Bitmap,这样就不会造成绘制很多路径,因为每次都绘制了全新的Bitmao
                //但是这样做的话,因为鼠标移动的次数是大量的,所以在内存中会创建大量的Bitmap会造成内存消耗严重,所以每次移动绘制完后,
                //需要释放Dispose() 画板,画笔,Bitmap资源。
                //
                Bitmap copyBtmp = (Bitmap)screenBtmp.Clone(); //创建新的,在其上绘制路径
                //左上角
                Point firstP = new Point(downPoint.X,downPoint.Y);
                //新建画板,画笔
                Graphics g = Graphics.FromImage(copyBtmp);
                Pen p = new Pen(Color.Red,1);
                //计算路径范围
                int width = Math.Abs(e.X - downPoint.X);
                int height = Math.Abs(e.Y - downPoint.Y);
                if (e.X < downPoint.X) {
                    firstP.X = e.X;
                }
                if (e.Y < downPoint.Y) {
                    firstP.Y = e.Y;
                }
                //绘制路径
                catchRec = new Rectangle(firstP,new Size(width,height));
                //将路径绘制在新的BItmap上,之后要释放
                g.DrawRectangle(p, catchRec);
                g.Dispose();
                p.Dispose();
 
                //窗体背景画板
                Graphics gf = this.CreateGraphics();
                //将新图绘制在窗体的画板上   --   自由截图-路径绘制处,其实还是一张和屏幕同样大小的图片,只不过上面有红色的选择路径
                gf.DrawImage(copyBtmp,new Point(0,0));
                gf.Dispose();
                //释放内存Bimtap
                copyBtmp.Dispose();
                
            }
        }
 
        bool catchFinished = false; //自由截图结束标志
        //鼠标左键弹起-结束自由截图
        private void Cutter_MouseUp(object sender, MouseEventArgs e) {
            if (e.Button == MouseButtons.Left) {
                if (CatchStart) {
                    CatchStart = false;
                    catchFinished = true;
                }
            }
        }
 
        //鼠标左键双击,保存自由截取的图片
        private void Cutter_MouseDoubleClick(object sender, MouseEventArgs e) {
            if((e.Button == MouseButtons.Left) && catchFinished){
                //创建用户截取的范围大小的空图
                Bitmap catchBtmp = new Bitmap(catchRec.Width,catchRec.Height);
                Graphics g = Graphics.FromImage(catchBtmp);
                //在原始的屏幕截图ScreenBitmap上 截取 用户选择范围大小的区域   绘制到上面的空图
                //绘制完后,这个空图就是我们想要的截取的图片
                //参数1  原图
                //参数2  在空图上绘制的范围区域
                //参数3  原图的截取范围
                //参数4  度量单位
                g.DrawImage(screenBtmp,new Rectangle(0,0,catchRec.Width,catchRec.Height),catchRec,GraphicsUnit.Pixel);
 
                //将自由截取的图片保存到剪切板中
                Clipboard.Clear();
                Clipboard.SetImage(catchBtmp);
                g.Dispose();
                catchFinished = false;
                this.BackgroundImage = screenBtmp;
                catchBtmp.Dispose();
                this.DialogResult = DialogResult.OK;
                this.Close();
            }
        }
    }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • C# xml序列化实现及遇到的坑

    C# xml序列化实现及遇到的坑

    在C#中,当我们需要将对象存储到文件或通过网络发送时,我们可以使用XML序列化将C#对象转换为XML文档,以便于存储、传输和还原,本文主要介绍了C# xml序列化实现及遇到的坑,感兴趣的可以了解一下
    2023-09-09
  • 详解WCF服务中的svc文件

    详解WCF服务中的svc文件

    本文详细讲解了WCF服务中的svc文件,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-03-03
  • 详解WPF如何显示具有层级关系的数据

    详解WPF如何显示具有层级关系的数据

    这篇文章主要为大家详细介绍了在WPF中如何显示具有层级关系的数据,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2024-04-04
  • 使用C# Winform应用程序获取网页源文件的解决方法

    使用C# Winform应用程序获取网页源文件的解决方法

    本篇文章是对使用C# Winform应用程序获取网页源文件的方法进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • C# MJPEG 客户端简单实现方法

    C# MJPEG 客户端简单实现方法

    这篇文章主要介绍了C# MJPEG 客户端简单实现的方法,帮助大家更好的理解和学习使用c#,感兴趣的朋友可以了解下
    2021-03-03
  • C# Socket 发送&接收&返回 简单应用实例

    C# Socket 发送&接收&返回 简单应用实例

    下面小编就为大家分享一篇C# Socket 发送&接收&返回 简单应用实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2017-11-11
  • C#操作windows系统进程的方法

    C#操作windows系统进程的方法

    这篇文章主要介绍了C#操作windows系统进程的方法,涉及C#针对windows操作系统进程的创建与关闭的技巧,需要的朋友可以参考下
    2015-04-04
  • C#中的==运算符

    C#中的==运算符

    这篇文章主要介绍了C#中的==运算符,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2017-06-06
  • WPF自定义实现雷达图控件的示例详解

    WPF自定义实现雷达图控件的示例详解

    雷达图用于表示不同内容的占比关系,在项目中有广泛的应用,但是目前未曾有封装良好的雷达图控件,所以本文分享了如何封装一个通用的雷达图控件,希望对大家有所帮助
    2023-08-08
  • c#反射表达式树模糊搜索示例

    c#反射表达式树模糊搜索示例

    这篇文章主要介绍了c#反射表达式树模糊搜索示例,反射实体T,非datetime字段反射获取表达式树,需要的朋友可以参考下
    2014-02-02

最新评论