?     LinearGradientBrush:使用沿渐变混合的两种颜色进行绘制

?     PathGradientBrush :基于编程者定义的唯一路径,使用复杂的混合色渐变进行绘制

我们这里只是简单介绍使用其中的几种:

Graphics g = this.CreateGraphics();

Rectangle rect = new Rectangle(10, 10, 50, 50);//定义矩形,参数为起点横纵坐标以及其长和宽

//单色填充

SolidBrush b1 = new SolidBrush(Color.Blue);//定义单色画刷

g.FillRectangle(b1, rect);//填充这个矩形

//字符串

g.DrawString("字符串", new Font("宋体", 10), b1, new PointF(90, 10));

//用图片填充

TextureBrush b2 = new TextureBrush(Image.FromFile(@"e:picture1.jpg"));

rect.Location = new Point(10, 70);//

rect.Width = 200;//更改这个矩形的宽来

rect.Height = 200;//更改这个矩形的高

g.FillRectangle(b2, rect);

//用渐变色填充

rect.Location = new Point(10, 290);

LinearGradientBrush b3 = new LinearGradientBrush(rect, Color.Yellow , Color.Black , LinearGradientMode.Horizontal);

g.FillRectangle(b3, rect);

3.坐标轴变换

在winform中的坐标轴和我们平时接触的平面直角坐标轴不同,winform中的坐标轴方向完全相反:窗体的左上角为原点(0,0),水平向左则X增大,垂直下向则Y增大

接下来,我们来实际操作下,通过旋转坐标轴的方向来画出不同角度的图案,或通过更改坐标原点的位置来平衡坐标轴的位置.

Graphics g = this.CreateGraphics();

//单色填充

//SolidBrush b1 = new SolidBrush(Color.Blue);//定义单色画刷

Pen p = new Pen(Color.Blue,1);

//转变坐标轴角度

for (int i = 0; i < 90; i++)

{

g.RotateTransform(i);//每旋转一度就画一条线

g.DrawLine(p, 0, 0, 100, 0);

g.ResetTransform();//恢复坐标轴坐标

}

//平移坐标轴

g.TranslateTransform(100, 100);

g.DrawLine(p, 0, 0, 100, 0);

g.ResetTransform();

//先平移到指定坐标,然后进行度旋转

g.TranslateTransform(100,200);

for (int i = 0; i < 8; i++)

{

g.RotateTransform(45);

g.DrawLine(p, 0, 0, 100, 0);

}

g.Dispose();

4.最后我们来看下Graphics这个画板上我们还可以画什么

其实我们上面用到的都是在画一些简单的图形,直线,矩形,扇形,圆孤等,我们还可以用它来绘制图片,这可以用它的DrawImage方法.这里我不详细讲解,大家有兴趣可以自己去MSDN了解下.我们后面会讲到的截图就会用到这个方法.

感谢大家的支持,这几天从早忙到晚,一个字累呀!!!现在挺困的,但是又不习惯这么早睡觉,哎~~还是利用这个时间继续来写第三篇吧.

前两篇已经基本向大家介绍了绘图的基本知识.那么,我就用我们上两篇所学的,做几个例子.

我们先来做一个简单的----仿QQ截图关于这个的例子其实网上已经有这方面的资料了,但是为了文章的完整性,还是觉得有必要讲解.

接下来看看这是如何做到的.

思路:聊天窗体上有一个截图按钮,点击按钮后,程序将整个屏幕画在一个新的全屏窗体上,然后显示这个窗体.因为是全屏的窗体,并且隐藏了菜单栏、工具栏等,所以在我们看来就好像是一个桌面的截图,然后在这个新窗体上画矩形,最后保存矩形中的内容并显示在原来的聊天窗体中.

步骤:

A.新建一个窗体.命名为Catch.然后设置这个窗体的FormBorderStyle为None,WindowState为Maximized.

B.我们对代码进行编辑:

using System;

using System.Collections.Generic;

using System.ComponentModel;

using System.Data;

using System.Drawing;

using System.Text;

using System.Windows.Forms;

namespace Client

{

public partial class Catch : Form

{

public Catch()

{

InitializeComponent();

}

用户变量#region 用户变量

private Point DownPoint = Point.Empty;//记录鼠标按下坐标,用来确定绘图起点

private bool CatchFinished = false;//用来表示是否截图完成

private bool CatchStart = false;//表示截图开始

private Bitmap originBmp;//用来保存原始图像

private Rectangle CatchRect;//用来保存截图的矩形

#endregion

//窗体初始化操作

private void Catch_Load(object sender, EventArgs e)

{

this.SetStyle(ControlStyles.OptimizedDoubleBuffer | ControlStyles.AllPaintingInWmPaint | ControlStyles.UserPaint, true);

this.UpdateStyles();

//以上两句是为了设置控件样式为双缓冲,这可以有效减少图片闪烁的问题,关于这个大家可以自己去搜索下

originBmp = new Bitmap(this.BackgroundImage);//BackgroundImage为全屏图片,我们另用变量来保存全屏图片

}

//鼠标右键点击结束截图

private void Catch_MouseClick(object sender, MouseEventArgs e)

{

if (e.Button == MouseButtons.Right)

{

this.DialogResult = DialogResult.OK;

this.Close();

}

}

//鼠标左键按下时动作

private void Catch_MouseDown(object sender, MouseEventArgs e)

{

if (e.Button == MouseButtons.Left)

{

if (!CatchStart)

{//如果捕捉没有开始

CatchStart = true;

DownPoint = new Point(e.X, e.Y);//保存鼠标按下坐标

}

}

}

private void Catch_MouseMove(object sender, MouseEventArgs e)

{

if (CatchStart)

{//如果捕捉开始

Bitmap destBmp = (Bitmap)originBmp.Clone();//新建一个图片对象,并让它与原始图片相同

Point newPoint = new Point(DownPoint.X, DownPoint.Y);//获取鼠标的坐标

Graphics g = Graphics.FromImage(destBmp);//在刚才新建的图片上新建一个画板

Pen p = new Pen(Color.Blue,1);

int width = 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 = new Rectangle(newPoint,new Size(width,height));//保存矩形

g.DrawRectangle(p,CatchRect);//将矩形画在这个画板上

g.Dispose();//释放目前的这个画板

p.Dispose();

Graphics g1 = this.CreateGraphics();//重新新建一个Graphics类

//如果之前那个画板不释放,而直接g=this.CreateGraphics()这样的话无法释放掉第一次创建的g,因为只是把地址转到新的g了.如同string一样

g1 = this.CreateGraphics();//在整个全屏窗体上新建画板

g1.DrawImage(destBmp,new Point(0,0));//将刚才所画的图片画到这个窗体上

//这个也可以属于二次缓冲技术,如果直接将矩形画在窗体上,会造成图片抖动并且会有无数个矩形.

g1.Dispose();

destBmp.Dispose();//要及时释放,不然内存将会被大量消耗

}

}

private void Catch_MouseUp(object sender, MouseEventArgs e)

{

if (e.Button == MouseButtons.Left)

{

if (CatchStart)

{

CatchStart = false;

CatchFinished = true;

}

}

}

//鼠标双击事件,如果鼠标位于矩形内,则将矩形内的图片保存到剪贴板中

private void Catch_MouseDoubleClick(object sender, MouseEventArgs e)

{

if (e.Button == MouseButtons.Left&&CatchFinished)

{

if (CatchRect.Contains(new Point(e.X, e.Y)))

{

Bitmap CatchedBmp = new Bitmap(CatchRect.Width, CatchRect.Height);//新建一个于矩形等大的空白图片

Graphics g = Graphics.FromImage(CatchedBmp);

g.DrawImage(originBmp, new Rectangle(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();

}

}

}

}

}

C.创建了Catch窗体后,我们在截图按钮(位于聊天窗体上)上加入以下事件:

private void bCatch_Click(object sender, EventArgs e)

{if (bCatch_HideCurrent.Checked)

{

this.Hide();//隐藏当前窗体

Thread.Sleep(50);//让线程睡眠一段时间,窗体消失需要一点时间

Catch CatchForm = new Catch();

Bitmap CatchBmp = new Bitmap(Screen.AllScreens[0].Bounds.Width, Screen.AllScreens[0].Bounds.Height);//新建一个和屏幕大小相同的图片

Graphics g = Graphics.FromImage(CatchBmp);

g.CopyFromScreen(new Point(0, 0), new Point(0, 0), new

Size(Screen.AllScreens[0].Bounds.Width,

Screen.AllScreens[0].Bounds.Height));//保存全屏图片

CatchForm.BackgroundImage = CatchBmp;//将Catch窗体的背景设为全屏时的图片

if (CatchForm.ShowDialog() == DialogResult.OK)

{//如果Catch窗体结束,就将剪贴板中的图片放到信息发送框中

IDataObject iData = Clipboard.GetDataObject();

DataFormats.Format myFormat = DataFormats.GetFormat(DataFormats.Bitmap);

if (iData.GetDataPresent(DataFormats.Bitmap))

{

richtextbox1.Paste(myFormat);

Clipboard.Clear();//清除剪贴板中的对象

}

this.Show();//重新显示窗体

}

}

}

这样我们的截图功能便完成了.

我想对于初学者来说如何消去第一次绘制的图片是个比较困难的问题.如果没有采取措施,你会发现只要你鼠标移动,就会画一个矩形,这样便会出现N多的矩形,而我们只是要最后的那一个.

一般解决这种问题的方法有两种:

1.就是在绘制第二个图形时,我们先用与底色相同的颜色将上次绘制的图形重新绘制一下.但这往往需要底色为纯色时使用.

2.我们并不直接将图形画在画板上,我们用一个图片A来保存原画板上的图片.然后再新建一个与图片A相同的图片B,将我们要绘制的图形画在该图片B上,然后再将该图片B画在画板上.这样图片A并没有被改变.于是第二次画的时候我们还是同样新建一个与图片A相同的图片进行绘制.那么上一次的图形就不会被保留下来.问题也就解决了.

下一次,向大家介绍如何做一个仿windows画板的程序.

前几篇我已经向大家介绍了如何使用GDI+来绘图,并做了一个截图的实例,这篇我向大家介绍下如何来做一个类似windows画图的工具.

主要实现功能:画直线,矩形,橡皮,圆形,切换颜色,打开图片,保存图片,清除图片,手动调节画布大小;软件刚启动时,为一张空白画布,我们可以直接在画布上绘图,也可以通过菜单中的“打开”,导入一张图片,然后我们就可以在这张图片上进行绘制。

平台:VS2005 WINFORM

由于代码过多,在这里只简要介绍下制作步骤,提供大家工程下载.

1.对整个界面进行布局.

2.实现绘图工具的功能

3.实现颜色拾取的功能,这里我们直接拿上次写的自定义控件来用.

4.实现菜单功能

5.实现手动调节画布大小的功能

6.测试

实现绘图工具的功能

为了让代码藕合度小点,稍许用了些设计模式,因为不是很会,所以代码还是有点乱乱的,绘图工具的这些功能块全部写在了DrawTools这个类里.那么在主窗体中,只需要调用这个类来完成绘制就行了,而不需要过多的涉及到具体的绘图代码。绘图工具这个类提供的主要工具就是:铅笔、橡皮、直线、矩形、圆形、实心矩形、实心圆形。关于这些功能块的代码,并不难,只要大家对认真看过前几篇内容,那应该都看得懂。

这里有几点要注意:

1.如何防止记录不必要的绘图过程中的痕迹?

这个问题在第三篇中有提到过,大家不妨先去看看那一篇。为了让代码看起来可读性高点,我设置了两个Image变量,finishingImg用来保存绘图过程中的痕迹,orginalImg用来保存已完成的绘图过程和初始时的背景图片。

2.这个类如何与主窗体进行通信?

当然如果直接将这些功能块写在主窗体中自然没有这个问题。但是那样代码会显得很混杂,如果只是工具代码出现问题就需要改整个项目。我在这里通过定义方法和属性,让主窗体通过给属性赋值将画板画布以及颜色什么的信息传给这个工具类,然后通过调用相应的工具方法来使用这些工具。

3.关键属性

要想让这些工具能正常使用,必须传递给他以下几样东西:目标画板(也就是picturebox),绘图颜色,原始画布。

实现菜单功能

这里就需要我们对文件的操作有一点了解,大家可以去查一下相关资料。

难点主要就是“打开”这个菜单项的实现

我们要实现将打开后的图片在修改后重新保存就必须让文件在打开后就能关闭,否则就会因为文件打开而无法覆盖原文件。就会导致编译时弹出“GDI  一般性错误”。所以根据网上其它朋友的做法就是先将打开的图片通过GDI+将图片画到另一个画布上,然后及时关闭打开的图片和用来绘制该图片的画板。详见http://www.wanxin.org/redirect.php?tid=3&goto=lastpost

private void openPic_Click(object sender, EventArgs e)

这里就需要我们对文件的操作有一点了解,大家可以去查一下相关资料。

难点主要就是“打开”这个菜单项的实现

我们要实现将打开后的图片在修改后重新保存就必须让文件在打开后就能关闭,否则就会因为文件打开而无法覆盖原文件。就会导致编译时弹出“GDI  一般性错误”。所以根据网上其它朋友的做法就是先将打开的图片通过GDI+将图片画到另一个画布上,然后及时关闭打开的图片和用来绘制该图片的画板。详见http://www.wanxin.org/redirect.php?tid=3&goto=lastpost

private void openPic_Click(object sender, EventArgs e)

{

OpenFileDialog ofd = new OpenFileDialog();//实例化文件打开对话框

ofd.Filter = "JPG|*.jpg|Bmp|*.bmp|所有文件|*.*";//设置对话框打开文件的括展名

if (ofd.ShowDialog() == DialogResult.OK)

{

Bitmap bmpformfile = new Bitmap(ofd.FileName);//获取打开的文件

panel2.AutoScrollPosition = new Point(0,0);//将滚动条复位

pbImg.Size = bmpformfile.Size;//调整绘图区大小为图片大小

reSize.Location = new Point(bmpformfile.Width, bmpformfile.Height);//reSize为我用来实现手动调节画布大小用的

//因为我们初始时的空白画布大小有限,"打开"操作可能引起画板大小改变,所以要将画板重新传入工具类

dt.DrawTools_Graphics = pbImg.CreateGraphics();

Bitmap bmp = new Bitmap(pbImg.Width, pbImg.Height);

Graphics g = Graphics.FromImage(bmp);

g.FillRectangle(new SolidBrush(pbImg.BackColor), new

Rectangle(0, 0, pbImg.Width, pbImg.Height));//不使用这句话,那么这个bmp的背景就是透明的

g.DrawImage(bmpformfile, 0, 0,bmpformfile.Width,bmpformfile.Height);//将图片画到画板上

g.Dispose();//释放画板所占资源

//不直接使用pbImg.Image = Image.FormFile(ofd.FileName)是因为这样会让图片一直处于打开状态,也就无法保存修改后的图片

bmpformfile.Dispose();//释放图片所占资源

g = pbImg.CreateGraphics();

g.DrawImage(bmp, 0, 0);

g.Dispose();

dt.OrginalImg = bmp;

bmp.Dispose();

sFileName = ofd.FileName;//储存打开的图片文件的详细路径,用来稍后能覆盖这个文件

ofd.Dispose();

}

}

清除图像其实就是用白色填充整个画布,其它的都比较简单,这就不具体讲了。

实现手动调节画布大小

网上有人说使用API,但是个人觉得还是使用其它控件帮忙比较简单,至少我们还看得懂。

思路:放置一个picturebox1(尺寸为5*5),将它固定在主画板的右下角,然后改变鼠标进入时的Cursor为箭头形状,设置鼠标按下移动时的事件,让该picturebox1 跟随鼠标移动。当鼠标松开时,将主画板的右下角坐标调整为picturebox1的坐标。

下面来看下代码:

其中的reSize就是我们用来帮忙的picturebox控件

private bool bReSize = false;//是否改变画布大小

private void reSize_MouseDown(object sender, MouseEventArgs e)

{

bReSize = true;//当鼠标按下时,说明要开始调节大小

}

private void reSize_MouseMove(object sender, MouseEventArgs e)

{

if (bReSize)

{

reSize.Location = new Point(reSize.Location.X + e.X, reSize.Location.Y + e.Y);

}

}

private void reSize_MouseUp(object sender, MouseEventArgs e)

{

bReSize = false;//大小改变结束

//调节大小可能造成画板大小超过屏幕区域,所以事先要设置autoScroll为true.

//但是滚动条的出现反而增加了我们的难度,因为滚动条上下移动并不会自动帮我们调整图片的坐标。

if (bReSize)

{

reSize.Location = new Point(reSize.Location.X + e.X, reSize.Location.Y + e.Y);

}

}

private void reSize_MouseUp(object sender, MouseEventArgs e)

{

bReSize = false;//大小改变结束

//调节大小可能造成画板大小超过屏幕区域,所以事先要设置autoScroll为true.

//但是滚动条的出现反而增加了我们的难度,因为滚动条上下移动并不会自动帮我们调整图片的坐标。

//这是因为GDI绘图的坐标系不只一个,好像有三个,没有仔细了解,一个是屏幕坐标,一个是客户区坐标,还个是文档坐标。

//滚动条的上下移动改变的是文档的坐标,但是客户区坐标不变,而location属性就属于客户区坐标,所以我们直接计算会出现错误

//这时我们就需要知道文档坐标与客户区坐标的偏移量,这就是AutoScrollPostion可以提供的

pbImg.Size = new Size(reSize.Location.X - (this.panel2.AutoScrollPosition.X), reSize.Location.Y - (this.panel2.AutoScrollPosition.Y));

dt.DrawTools_Graphics = pbImg.CreateGraphics();//因为画板的大小被改变所以必须重新赋值

//另外画布也被改变所以也要重新赋值

Bitmap bmp = new Bitmap(pbImg.Width, pbImg.Height);

Graphics g = Graphics.FromImage(bmp);

g.FillRectangle(new SolidBrush(Color.White), 0, 0, pbImg.Width, pbImg.Height);

g.DrawImage(dt.OrginalImg, 0, 0);

g.Dispose();

g = pbImg.CreateGraphics();

g.DrawImage(bmp, 0, 0);

g.Dispose();

dt.OrginalImg = bmp;

bmp.Dispose();

}

此时就可以通过拖动那个小方块来调节图片大小了。

c#创建画布_C#中的绘图相关推荐

  1. c#创建画布_c# GDI+简单绘图(四) 简易画板功能

    前几篇我已经向大家介绍了如何使用GDI+来绘图,并做了一个截图的实例,这篇我向大家介绍下如何来做一个类似windows画图的工具. 个人认为如果想做一个功能强大的绘图工具,那么单纯掌握GDI还远远不够 ...

  2. c#创建画布_C#GDI+编程基础(一:Graphics画布类)

    GDI+存在的意义:将变成与具体硬件实现细节分开. GDI+步骤:获取画布,绘制图像.处理图像 命名空间: using System.Drawing;//提供对GDI+基本图形功能的访问 using ...

  3. c#创建画布_C# – 调整图像画布大小(保留源图像的原始像素尺寸)

    我的目标是拍摄一个图像文件,并将尺寸增加到下一次的两倍,同时保持像素(也就是不缩放源图像).因此,基本上最终的结果将是原始图像,加上跨越图像右下角的额外的白色空间,因此总尺寸是二的幂. 下面是我现在使 ...

  4. arcgis mxt模板 创建工具条无法保存_【从零开始学GIS】ArcGIS中的绘图基本操作(二)...

    大家好,我是肝教程肝到熊猫眼的三三. 本系列教程的发布,受到了很多同学的鼓励,大家在后台或微信上表达出对教程的喜爱,这便是更新教程的最大动力. 上回教程讲解了"GIS基本操作".& ...

  5. python创建画布_Python——绘图

    1.保存图片. fig.savefig 一.创建画布 1.创建画布和坐标轴 在Matplotlib中,plt.figure类可以看做一个能够容纳各种坐标轴.图形.文字和标签的容器.plt.Axes类是 ...

  6. HTML5怎样创建画布?

    在网页中,画布并不是默认存在的,用户首先需要创建画布,然后通过一些对象和方法可以在画布中绘制图案,下面将分步骤讲解使用画布的方法. 1. 创建画布 使用HTML5中的canvas标签可以在网页中创建画 ...

  7. Android中Canvas绘图之Shader使用图文详解

    概述 我们在用Android中的Canvas绘制各种图形时,可以通过Paint.setShader(shader)方法为画笔Paint设置shader,这样就可以绘制出多彩的图形.那么Shader是什 ...

  8. 初始化创建画布_使用HTML5,画布和开放数据创建全球降水(雨)可视化

    初始化创建画布 我目前正在为Three.js编写下一本书,其中一章涉及可视化开放数据. 在寻找可以使用的数据时,我遇到了来自NOAA的一组数据. 通过此站点,您可以以网格格式下载一组全世界的每月降水报 ...

  9. python绘图背景透明_如何在 Matplotlib 中更改绘图背景

    介绍Matplotlib是Python中使用最广泛的数据可视化库之一.无论是简单还是复杂的可视化项目,它都是大多数人的首选库.在本教程中,我们将研究如何在Matplotlib中更改绘图的背景.导入数据 ...

最新文章

  1. 使用NeMo快速入门NLP、实现机器翻译任务,英伟达专家实战讲解,内附代码
  2. 【必看】 一篇 CPU 占用高,导致请求超时的故障排查
  3. 通过jS打包下载图片
  4. Linux脚本:xjps查看各个节点java进程
  5. 实验十一:图形界面二
  6. 在PHP中如何要json中的数据,如何在不知道键值的情况下在php中读取JSON数据
  7. 轻松搞定对容器实例日志设置定期清理和回卷 1
  8. html5学习笔记(7)
  9. 关于各类图形CAD底层内核
  10. 电脑计算机网络都打不开怎么办,电脑网页打不开怎么回事的常规处理方法
  11. 按键精灵安卓版去除重复数组然后排序排序
  12. ArcGIS Maps SDK for Unreal Engine通过UI方式显示地图教程
  13. python自动生成和读取word_使用Python自动生成Word文档的教程
  14. Promise的理解
  15. BMI(体重指数)计算C语言
  16. 在华为五个月,我的所见所得!
  17. 2016太原网络营销师揭秘面试题百度竞价(SEM)中百度网盟推广的理解?
  18. luoguP4466 [国际集训队]和与积 莫比乌斯反演
  19. HBase问题诊断 – RegionServer宕机
  20. 将批处理文件编译成可执行文件

热门文章

  1. mybatis注解开发 @Delete 批量删除
  2. 驳马德兴《国奥输给裁判》:的确技不如人
  3. 结对编程 电梯调度程序
  4. 三个练手的软件测试实战项目(附全套视频跟源码)偷偷卷死他们
  5. 微信小程序获取用户信息接口突然不弹授权框了,用户名‘微信用户’,头像空白(<button open-type=“getUserInfo“/>已弃用)
  6. Es6新特性:set
  7. 3.html实现网页布局
  8. 跟着Cell Research学单细胞分析:以小提琴图、热图为例
  9. 在蚂蚁实习的笔试面试经验全公开,也拿过鹅头多offer
  10. 方舟服务器建家位置,方舟建家位置