深入c#GDI+简单绘图的具体操作步骤(三)
c# 简单 深入 具体 绘图 操作步骤 gdi
2023-06-13 09:14:54 时间
关于这个的例子其实网上已经有这方面的资料了,但是为了文章的完整性,还是觉得有必要讲解.
我们先来看一下效果:
(图(图1)
(图2)
接下来看看这是如何做到的.
思路:聊天窗体上有一个截图按钮,点击按钮后,程序将整个屏幕画在一个新的全屏窗体上,然后显示这个窗体.因为是全屏的窗体,并且隐藏了菜单栏、工具栏等,所以在我们看来就好像是一个桌面的截图,然后在这个新窗体上画矩形,最后保存矩形中的内容并显示在原来的聊天窗体中.
步骤:
A.新建一个窗体.命名为Catch.然后设置这个窗体的FormBorderStyle为None,WindowState为Maximized.
B.我们对代码进行编辑:
复制代码代码如下:
我们先来看一下效果:
(图(图1)
(图2)
步骤:
B.我们对代码进行编辑:
usingSystem;
usingSystem.Collections.Generic;
usingSystem.ComponentModel;
usingSystem.Data;
usingSystem.Drawing;
usingSystem.Text;
usingSystem.Windows.Forms;
namespaceClient
{
publicpartialclassCatch:Form
{
publicCatch()
{
InitializeComponent();
}
用户变量#region用户变量
privatePointDownPoint=Point.Empty;//记录鼠标按下坐标,用来确定绘图起点
privateboolCatchFinished=false;//用来表示是否截图完成
privateboolCatchStart=false;//表示截图开始
privateBitmaporiginBmp;//用来保存原始图像
privateRectangleCatchRect;//用来保存截图的矩形
#endregion
//窗体初始化操作
privatevoidCatch_Load(objectsender,EventArgse)
{
this.SetStyle(ControlStyles.OptimizedDoubleBuffer|ControlStyles.AllPaintingInWmPaint|ControlStyles.UserPaint,true);
this.UpdateStyles();
//以上两句是为了设置控件样式为双缓冲,这可以有效减少图片闪烁的问题,关于这个大家可以自己去搜索下
originBmp=newBitmap(this.BackgroundImage);//BackgroundImage为全屏图片,我们另用变量来保存全屏图片
}
//鼠标右键点击结束截图
privatevoidCatch_MouseClick(objectsender,MouseEventArgse)
{
if(e.Button==MouseButtons.Right)
{
this.DialogResult=DialogResult.OK;
this.Close();
}
}
//鼠标左键按下时动作
privatevoidCatch_MouseDown(objectsender,MouseEventArgse)
{
if(e.Button==MouseButtons.Left)
{
if(!CatchStart)
{//如果捕捉没有开始
CatchStart=true;
DownPoint=newPoint(e.X,e.Y);//保存鼠标按下坐标
}
}
}
privatevoidCatch_MouseMove(objectsender,MouseEventArgse)
{
if(CatchStart)
{//如果捕捉开始
BitmapdestBmp=(Bitmap)originBmp.Clone();//新建一个图片对象,并让它与原始图片相同
PointnewPoint=newPoint(DownPoint.X,DownPoint.Y);//获取鼠标的坐标
Graphicsg=Graphics.FromImage(destBmp);//在刚才新建的图片上新建一个画板
Penp=newPen(Color.Blue,1);
intwidth=Math.Abs(e.X-DownPoint.X),height=Math.Abs(e.Y-DownPoint.Y);//获取矩形的长和宽
if(e.X<DownPoint.X)
{
newPoint.X=e.X;
}
if(e.Y<DownPoint.Y)
{
newPoint.Y=e.Y;
}
CatchRect=newRectangle(newPoint,newSize(width,height));//保存矩形
g.DrawRectangle(p,CatchRect);//将矩形画在这个画板上
g.Dispose();//释放目前的这个画板
p.Dispose();
Graphicsg1=this.CreateGraphics();//重新新建一个Graphics类
//如果之前那个画板不释放,而直接g=this.CreateGraphics()这样的话无法释放掉第一次创建的g,因为只是把地址转到新的g了.如同string一样
g1=this.CreateGraphics();//在整个全屏窗体上新建画板
g1.DrawImage(destBmp,newPoint(0,0));//将刚才所画的图片画到这个窗体上
//这个也可以属于二次缓冲技术,如果直接将矩形画在窗体上,会造成图片抖动并且会有无数个矩形.
g1.Dispose();
destBmp.Dispose();//要及时释放,不然内存将会被大量消耗
}
}
privatevoidCatch_MouseUp(objectsender,MouseEventArgse)
{
if(e.Button==MouseButtons.Left)
{
if(CatchStart)
{
CatchStart=false;
CatchFinished=true;
}
}
}
//鼠标双击事件,如果鼠标位于矩形内,则将矩形内的图片保存到剪贴板中
privatevoidCatch_MouseDoubleClick(objectsender,MouseEventArgse)
{
if(e.Button==MouseButtons.Left&&CatchFinished)
{
if(CatchRect.Contains(newPoint(e.X,e.Y)))
{
BitmapCatchedBmp=newBitmap(CatchRect.Width,CatchRect.Height);//新建一个于矩形等大的空白图片
Graphicsg=Graphics.FromImage(CatchedBmp);
g.DrawImage(originBmp,newRectangle(0,0,CatchRect.Width,CatchRect.Height),CatchRect,GraphicsUnit.Pixel);
//把orginBmp中的指定部分按照指定大小画在画板上
Clipboard.SetImage(CatchedBmp);//将图片保存到剪贴板
g.Dispose();
CatchFinished=false;
this.BackgroundImage=originBmp;
CatchedBmp.Dispose();
this.DialogResult=DialogResult.OK;
this.Close();
}
}
}
}
}
privatevoidbCatch_Click(objectsender,EventArgse)
{
if(bCatch_HideCurrent.Checked)
{
this.Hide();//隐藏当前窗体
Thread.Sleep(50);//让线程睡眠一段时间,窗体消失需要一点时间
CatchCatchForm=newCatch();
BitmapCatchBmp=newBitmap(Screen.AllScreens[0].Bounds.Width,Screen.AllScreens[0].Bounds.Height);//新建一个和屏幕大小相同的图片
Graphicsg=Graphics.FromImage(CatchBmp);
g.CopyFromScreen(newPoint(0,0),newPoint(0,0),newSize(Screen.AllScreens[0].Bounds.Width,Screen.AllScreens[0].Bounds.Height));//保存全屏图片
CatchForm.BackgroundImage=CatchBmp;//将Catch窗体的背景设为全屏时的图片
if(CatchForm.ShowDialog()==DialogResult.OK)
{//如果Catch窗体结束,就将剪贴板中的图片放到信息发送框中
IDataObjectiData=Clipboard.GetDataObject();
DataFormats.FormatmyFormat=DataFormats.GetFormat(DataFormats.Bitmap);
if(iData.GetDataPresent(DataFormats.Bitmap))
{
richtextbox1.Paste(myFormat);
Clipboard.Clear();//清除剪贴板中的对象
}
this.Show();//重新显示窗体
}
}
}
这样我们的截图功能便完成了.
我想对于初学者来说如何消去第一次绘制的图片是个比较困难的问题.如果没有采取措施,你会发现只要你鼠标移动,就会画一个矩形,这样便会出现N多的矩形,而我们只是要最后的那一个.
2.我们并不直接将图形画在画板上,我们用一个图片A来保存原画板上的图片.然后再新建一个与图片A相同的图片B,将我们要绘制的图形画在该图片B上,然后再将该图片B画在画板上.这样图片A并没有被改变.于是第二次画的时候我们还是同样新建一个与图片A相同的图片进行绘制.那么上一次的图形就不会被保留下来.问题也就解决了.
相关文章
- unity c#面试_spring面试题及答案
- C#中File和FileStream的用法
- 【说站】C#在PDF中添加墨迹注释Ink Annotation的步骤详解
- 一个C#开发者学习SpringCloud搭建微服务的心路历程
- C#-ToolTIp和Popup简单使用
- dotnet6 C# 一个国内还能用的 NTP 时间校准客户端的实现
- C#没有动态的数组,可以用arraylist或list取代
- c#网址压缩简单实现短网址
- c#遍历System.drawing.Color下面的所有颜色以及名称以查看
- c#简单工厂、工厂方法与抽象工厂的区别分析
- C#仿密保卡功能的简单实现代码
- 深入c#GDI+简单绘图的具体操作步骤(一)
- C#interface与delegate效能比较的深入解析
- C#全角和半角转换以及判断的简单代码
- c#控件截图的简单实例
- C#Js时间格式化问题简单实例
- C#Access数据库增删查改的简单方法
- C#判断当前程序是否通过管理员运行的方法
- C#省份城市下拉框联动简单实现方法
- C#实现的简单验证码识别实例
- 浅谈C#设计模式之工厂模式
- 基于C#实现一个最简单的HTTP服务器实例
- C#实现根据数字序号输出星期几的简单实例
- C#中使用1.7版本驱动操作MongoDB简单例子
- C#生成条形码图片的简单方法
- C#中winform使用相对路径读取文件的方法
- C#实现线程池的简单示例
- C#函数式编程中的递归调用之尾递归详解
- C#实现复杂XML的序列化与反序列化