html5 Canvas画图教程(2)—画直线与设置线条的样式如颜色/端点/交汇点


Posted in HTML / CSS onJanuary 09, 2013

如果你还不知道Canvas是什么,可以看看上一篇.
在学画画的时候,线条是最基本的了,而线条的连接可以组成任何图形。在Canvas中也是如此。
在开始之前我们先拿出画布和画笔:

复制代码
代码如下:

var cvs = document.getElementById('cvs'); //画布
var ctx = cvs.getContext('2d'); // 画笔

我们画画的时候,落笔点是不固定的,随时都会变。canvas虽然不是通过手来决定落笔点,但也有一个方法,就是moveTo。moveTo的作用相当于把笔尖提离画布,然后移动到指定的点(即坐标)。
复制代码
代码如下:

ctx.moveTo(x,y)

此过程中不会画出任何图形,相当于你提着笔在画布上晃来晃去。
但晃来晃去是没用的,我们必须开始画。先画最简单的:直线
画直线的方法即lineTo,他的参数和moveTo一样的,都是一个点。
ctx.lineTo(x,y)当然,你画线的时候,落笔点也跟着移动了,所以lineTo之后落笔点就变成了他的目标点了。
复制代码
代码如下:

ctx.moveTo(100,100);
ctx.lineTo(200,100);

此时你刷新网页,会发现画布上没有预想中的线,什么也没有。因为我们还少了一个步骤.lineTo其实是画的一条“路径”,本身是不可见的。如果要让他显示出来,我们必须对他进行“画”的操作。
用过PS的同学,肯定能知道图形的两种模式,一种是填充,另一种是描边。现在我们已经画了一条线,相当于PS中勾了一条路径,此时给路径描一下边,就能显示出图形了。
canvas描边的方法是stroke().现在让我们把代码补全:
复制代码
代码如下:

ctx.moveTo(100,100);
ctx.lineTo(200,100);

ctx.stroke();此时刷新,就能看到一条线了。当然,你也可以连续绘制出几百条路径,再执行描边动作,即可一下画出几百条线。现在我们来用4条线画一个矩形:
复制代码
代码如下:

ctx.moveTo(100,100);
ctx.lineTo(200,100);
ctx.lineTo(200,200);
ctx.lineTo(100,200);
ctx.lineTo(100,100);
ctx.stroke();

这里我们就是先绘出全部路径,再一次性描边。
——–作者的抱怨:Canvas绘图有个不好的地方就是:基本靠猜,很不直观。
严重提示:canvas的绘图过程(即填充与描边)是非常消耗资源的,如果想节省系统资源提高效率,最好是绘制好所有路径,再一次性填充或描边图形。
由上面的图形我们可以看出,默认的线条粗细是1px,而线条颜色是黑色。当然我们可以设置他们,但奇怪的是设置线条宽度是lineWidth,而设置线条样式的却叫strokeStyle,为什么不是lineStyle呢?我也不知道。:
复制代码
代码如下:

ctx.lineWidth = 10;
ctx.strokeStyle = 'rgba(255,0,0,0.5)';

上面的代码把线条宽度设置成了10px,线条颜色变成了半透明的红色。

html5 Canvas画图教程(2)—画直线与设置线条的样式如颜色/端点/交汇点

如图1,刷新一下,好像有点不对呢!怎么左上角缺了一小块似得?这不是错觉。原因嘛要从canvas的线条绘制方式说起。
问题:如果我画的矩形路径是宽高都是100,此时我的边线宽度是10px,那么这个描了边的矩形整体宽高是多少?是100+10*2=120吗?
如果边线是完全描在路径外侧的话,那么就是120。但Canvas不是。Canvas中的线条都有一条“中线”,这条中线位于line的绝对中间,线条的笔画以中线向两侧扩展。比如你的线条宽度是1,那么中线就在0.5的位置;宽度是5,那么中线就是在2.5。canvas的图形在描边的时候,都是路径与line的中线贴合,然后描边。如图2:

html5 Canvas画图教程(2)—画直线与设置线条的样式如颜色/端点/交汇点


所以,描边的时候线的一半在外侧,一半在内侧,即上面的矩形的整体宽度是100+(10/2)*2,等于110.
也正是因为这个原因,左上角出现缺角就理所当然了。因为这里没人画。
但为什么其余的角没有出现缺口呢?看你的图不是4个角都有缺口吗?
那是因为,我画线的过程中没有把画笔“提起来”,画笔是连续的,即没有moveTo过。不信我们现在来moveTo一下:
复制代码
代码如下:

ctx.moveTo(100,100);
ctx.lineTo(200,100);
ctx.moveTo(200,100); //注意这里
ctx.lineTo(200,200);
ctx.lineTo(100,200);
ctx.lineTo(100,100);
ctx.lineWidth = 10;
ctx.strokeStyle = 'rgba(255,0,0,0.5)';
ctx.stroke();

我们再画第二条线之前moveTo了一下,而且moveTo连坐标都没变,还是那个点,但刷新后图形变成了这个样子[图3]:

html5 Canvas画图教程(2)—画直线与设置线条的样式如颜色/端点/交汇点


明白了?因为我们把笔提起来了。
现在我们删掉moveTo,不要纠结他了,我们来思考一下如何把左上角那个缺角给补上?
首先问个问题,我们的路径闭合了吗?这不是废话么,我们不是已经把路径绕回原点了么?当然算是闭合了!
错!这样只是让路径最后一个点和起点重合了而已,路径本身却没有闭合!
Canvas怎么闭合路径?用closePath().
复制代码
代码如下:

ctx.moveTo(100,100);
ctx.lineTo(200,100);
ctx.lineTo(200,200);
ctx.lineTo(100,200);
ctx.lineTo(100,100);
ctx.closePath();//闭合路径
ctx.lineWidth = 10;
ctx.strokeStyle = 'rgba(255,0,0,0.5)';
ctx.stroke();

此时刷新,就是一个完美的正方形了。图4:

html5 Canvas画图教程(2)—画直线与设置线条的样式如颜色/端点/交汇点


无论我们把线条改到多粗————越粗越有人喜欢是吧?————这个四方形的四个角都是规矩的直角,不会出现圆滑的情况。圆滑的角是什么情况?请看PS中的四方形描边,图5:

html5 Canvas画图教程(2)—画直线与设置线条的样式如颜色/端点/交汇点


看到了吧,越粗的边线,他的角的圆弧越大。
如果我想Canvas里面的边线也和PS这种一样,有没有办法呢?当然有,就是lineJoin属性。
lineJoin,意思即线的交汇处,有3个属性:miter(默认,尖角),bevel(斜角),round(圆角),如图6:
html5 Canvas画图教程(2)—画直线与设置线条的样式如颜色/端点/交汇点 
毫无疑问我们一下就能明白我们的矩形用的是尖角,所以试着把他改成圆角看看:
图形变成了这样,图7:
html5 Canvas画图教程(2)—画直线与设置线条的样式如颜色/端点/交汇点 
有点像PS的了吧?
另外,通过前面图我们了解到,Canvas的线条两端是平的,可不可以改呢?毕竟平的不好看。
也是可以的,即lineCap属性,这个就是定义线条的端点。lineCap有3个值:butt(平,默认),round(圆),square(方),如图8
html5 Canvas画图教程(2)—画直线与设置线条的样式如颜色/端点/交汇点 
看图就能发现,其实平头跟方头是一样的,区别只是平头没有伸出去那么一截。圆头和方头都会伸出去一截,这一节是多长呢?就是线条宽度的一半。
你有没有想到什么?哈哈,前面的闭合路径的问题,如果我们把lineCap设为方头,效果也是一样的!
但为了保险起见,我们还是要把路径闭合了,切记!
我还要提醒一下:闭合的路径没有端点!所以闭合的路径上看不到端点的样式。
另外:lineCap与lineJoin有点相似,注意不要搞混。
如果你眼尖并且运气不好,你可能会发现有时候1像素的线条不是1像素宽,好像要宽一些,模糊一些。如图9:
html5 Canvas画图教程(2)—画直线与设置线条的样式如颜色/端点/交汇点
恭喜你!你遇到了一个不是bug的bug。这个很特别,我把他放到下一篇文章讲吧
HTML / CSS 相关文章推荐
CSS心形加载的动画源码的实现
Mar 09 HTML / CSS
image-set实现Retina屏幕下图片显示详细介绍
Dec 24 HTML / CSS
html5中监听canvas内部元素点击事件的三种方法
Apr 28 HTML / CSS
HTML5 Canvas 起步(1) - 基本概念
May 12 HTML / CSS
使用jquery实现HTML5响应式导航菜单教程
Apr 02 HTML / CSS
谈一谈HTML5本地存储技术
Mar 02 HTML / CSS
HTML5声音录制/播放功能的实现代码
May 03 HTML / CSS
canvas中普通动效与粒子动效的实现代码示例
Jan 03 HTML / CSS
Canvas环形饼图与手势控制的实现代码
Nov 08 HTML / CSS
HTML5调用手机发短信和打电话功能
Apr 29 HTML / CSS
CSS3 制作精美的定价表
Apr 06 HTML / CSS
html中相对位置与绝对位置的具体使用
May 15 HTML / CSS
html5 Canvas画图教程(1)—画图的基本常识
Jan 09 #HTML / CSS
使用html5+css3来实现slider切换效果告别javascript+css
Jan 08 #HTML / CSS
几个解决兼容IE6\7\8不支持html5标签的几个方法
Jan 07 #HTML / CSS
html5 canvas-2.用canvas制作一个猜字母的小游戏
Jan 07 #HTML / CSS
html5 canvas-1.canvas介绍(hello canvas)
Jan 07 #HTML / CSS
html5指南-1.html5全局属性(html5 global attributes)深入理解
Jan 07 #HTML / CSS
html5指南-2.如何操作document metadata
Jan 07 #HTML / CSS
You might like
NOT NULL 和NULL
2007/01/15 PHP
php入门教程 精简版
2009/12/13 PHP
PHP实现AES256加密算法实例
2014/09/22 PHP
Laravel 5框架学习之日期,Mutator 和 Scope
2015/04/08 PHP
php生成二维码不保存服务器还有下载功能的实现代码
2018/08/09 PHP
Laravel自动生成UUID,从建表到使用详解
2019/10/24 PHP
用jscript实现新建word文档
2007/06/15 Javascript
JQuery团队打造的javascript单元测试工具QUnit介绍
2010/02/26 Javascript
动态加载脚本提升javascript性能
2014/02/24 Javascript
Nodejs中的this详解
2016/03/26 NodeJs
JavaScript之Vue.js【入门基础】
2016/12/06 Javascript
浅述节点的创建及常见功能的实现
2016/12/15 Javascript
Angular2 PrimeNG分页模块学习
2017/01/14 Javascript
create-react-app构建项目慢的解决方法
2018/03/14 Javascript
JS实现统计字符串中字符出现个数及最大个数功能示例
2018/06/04 Javascript
vue-resource请求实现http登录拦截或者路由拦截的方法
2018/07/11 Javascript
VUE的history模式下除了index外其他路由404报错解决办法
2019/08/21 Javascript
简单了解JavaScript sort方法
2019/11/25 Javascript
在nodejs中创建child process的方法
2021/01/26 NodeJs
vue-router懒加载的3种方式汇总
2021/02/28 Vue.js
[52:03]DOTA2-DPC中国联赛 正赛 Ehome vs iG BO3 第三场 1月31日
2021/03/11 DOTA
使用django-suit为django 1.7 admin后台添加模板
2014/11/18 Python
Python中的fileinput模块的简单实用示例
2015/07/09 Python
Python实现基于二叉树存储结构的堆排序算法示例
2017/12/08 Python
Python静态类型检查新工具之pyright 使用指南
2019/04/26 Python
解决django 新增加用户信息出现错误的问题
2019/07/28 Python
Django中create和save方法的不同
2019/08/13 Python
python 哈希表实现简单python字典代码实例
2019/09/27 Python
Python入门基础之数字字符串与列表
2021/02/01 Python
localStorage 设置过期时间的方法实现
2018/12/21 HTML / CSS
ALDI奥乐齐官方海外旗舰店:德国百年超市
2017/12/27 全球购物
香蕉共和国工厂店:Banana Republic Factory
2018/06/09 全球购物
会议邀请书范文
2014/02/02 职场文书
人力资源管理求职信
2014/08/07 职场文书
部门2014年度工作总结
2014/11/12 职场文书
golang生成vcf通讯录格式文件详情
2022/03/25 Golang