接上一篇写的截取电脑屏幕,我们在原来的基础上加一个选择区域的功能,实现自定义选择截图。 个人比较懒,上一篇的代码就不重新设计了,就简单改一下呈现方式。 不得不吐槽一下,在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功能,直接将截图中的数据转化为文字。 由简入繁,拿来即用 后续精彩,持续关注
|