C# 简单实现QQ截图功能

这篇博客介绍了如何在Windows10环境下,通过修改现有代码实现屏幕选择区域的自定义截图功能。开发者使用C#在Visual Studio 2013和.NET Framework 4.5环境下,通过DLL引用和图形处理技术,实现了DPI缩放比例的计算,以适应不同屏幕放大比例的系统。用户可以选择截图区域,并将截图保存到剪切板。此外,博客还预告了后续将介绍的文字识别功能。
摘要由CSDN通过智能技术生成

接上一篇写的截取电脑屏幕,我们在原来的基础上加一个选择区域的功能,实现自定义选择截图。

个人比较懒,上一篇的代码就不重新设计了,就简单改一下呈现方式。

不得不吐槽一下,在windows10系统上设置了放大比例的话,用这种方式来实现截图功能的话需要去计算比例。后面有机会的话,用第三方DLL再实现一次。

实现功能:

屏幕选择区域截图

开发环境:

开发工具: Visual Studio 2013

.NET Framework版本:4.5

实现代码:

 

//将上一篇的内容改成以下内容
// pictureBox1.Image = bmp;
Form2 frm = new Form2();
frm.BaseImage = bmp;
frm.TopMost = true;
frm.Show();
/*Form2代码*/
#region Dll引用
[DllImport("User32.dll", EntryPoint = "GetDC")]
private extern static IntPtr GetDC(IntPtr hWnd);

[DllImport("User32.dll", EntryPoint = "ReleaseDC")]
private extern static int ReleaseDC(IntPtr hWnd, IntPtr hDC);

[DllImport("gdi32.dll")]
public static extern int GetDeviceCaps(IntPtr hdc, int nIndex);

[DllImport("User32.dll")]
public static extern int GetSystemMetrics(int hWnd);

const int DESKTOPVERTRES = 117;
const int DESKTOPHORZRES = 118;

const int SM_CXSCREEN = 0;
const int SM_CYSCREEN = 1;

#endregion

//<summary>
//获取DPI缩放比例
//</summary>
//<param name="dpiscalex"></param>
//<param name="dpiscaley"></param>
public static void GetDPIScale(ref float dpiscalex, ref float dpiscaley)
{
    int x = GetSystemMetrics(SM_CXSCREEN);
    int y = GetSystemMetrics(SM_CYSCREEN);
    IntPtr hdc = GetDC(IntPtr.Zero);
    int w = GetDeviceCaps(hdc, DESKTOPHORZRES);
    int h = GetDeviceCaps(hdc, DESKTOPVERTRES);
    ReleaseDC(IntPtr.Zero, hdc);
    dpiscalex = (float)w / x;
    dpiscaley = (float)h / y;
}


public Bitmap BaseImage { get; set; }

Graphics picGraphics;
//记录鼠标开始截图的位置
int startX = 0, startY = 0;
//记录鼠标结束截图的位置
int endX = 0, endY = 0;
//记录DPI缩放比例
float x = 0f, y = 0f;
public Form2()
{
    InitializeComponent();
}
private void Form2_Load(object sender, EventArgs e)
{
    /* 初始化赋值*/

    GetDPIScale(ref x, ref y);
    picGraphics = pictureBox1.CreateGraphics();
}

private void pictureBox1_Paint(object sender, PaintEventArgs e)
{
    DrawRect(endX - startX, endY - startY,e.Graphics);
}

private void Form2_KeyPress(object sender, KeyPressEventArgs e)
{
    //按下esc退出
    if (e.KeyChar == 27)
    {
        this.Close();
    }
}

//完成截图
private void lbSucess_Click(object sender, EventArgs e)
{
    int clip_w = (int)((endX - startX) * x) - 4, clip_h = (int)((endY - startY) * y) - 4;
    if (clip_w < 1 || clip_h < 1)
    {
        return;
    }
    //获取截图
    Bitmap clipBmp = new Bitmap(clip_w, clip_h);
    Graphics g = Graphics.FromImage(clipBmp);
    g.CopyFromScreen((int)(startX * x) + 2, (int)(startY * y) + 2, 0, 0, new Size(clip_w, clip_h), CopyPixelOperation.SourceCopy);
    //将截图设置到剪切板
    Clipboard.SetImage(clipBmp);
    g.Dispose();
    this.Close();
}

private void pictureBox1_MouseDown(object sender, MouseEventArgs e)
{
    //隐藏操作面板
    panel1.Visible = false;
    //记录鼠标开始截图的位置
    if (e.Button == MouseButtons.Left)
    {
        startX = e.X;
        startY = e.Y;
        endX = 0; endY = 0;
    }
}

private void pictureBox1_MouseMove(object sender, MouseEventArgs e)
{
    if (e.Button == MouseButtons.Left)
    {
        //绘制截图区域
        DrawRect(e.X - startX, e.Y - startY, picGraphics);
    }
}

private void pictureBox1_MouseUp(object sender, MouseEventArgs e)
{
    //截图完成
    if (e.Button == MouseButtons.Left)
    {
        endX = e.X;
        endY = e.Y;

        DrawRect(endX - startX, endY - startY, picGraphics);
        if (endX > startX && endY > startY)
        {
            //显示操作面板
            Thread.Sleep(100);
            panel1.Location = new Point(e.X - panel1.Width, e.Y + 5);
            panel1.Visible = true;
           
        }
    }
}

//绘制截图区域
private void DrawRect(int w, int h,Graphics g)
{
    Bitmap img = (Bitmap)BaseImage.Clone();

    //双缓冲技术画矩形,防止重影和抖动
    Graphics Painter = Graphics.FromImage(img);
    Painter.DrawRectangle(new Pen(Color.Red), startX * x, startY * y, w * x, h * y);
    g.DrawImage(img, 0, 0, img.Width / x, img.Height / y);

    Painter.Dispose();
    img.Dispose();
}

实现效果:

我这边演示代码很少做异常处理(前面也是,后面也是,毕竟不是做项目为主),大家使用的时候根据情况自行处理下即可,亦或者可能会有内存未及时释放的情况。

有朋友可能也看到演示效果中的另一个选项是文字识别,嗯,下一篇写OCR功能,直接将截图中的数据转化为文字。

由简入繁,拿来即用

更多精彩,请关注微信公众号:

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值