【基础系列】Canvas专题

1 Canvas接口元素定义

1.1 getContext()方法

        为了在canvas上绘制,你必须先得到一个画布上下文对象的引用,用本方法即可完成这一操作,格式如下:

context = canvas . getContext(contextId)

        方法返回一个指定contextId的上下文对象,如果指定的id不被支持,则返回null,当前唯一被强制必须支持的是“2d”,也许在将来会有“3d”,注意,指定的id是大小写敏感的。

1.2 toDataURL()方法

        返回一张使用canvas绘制的图片,返回值符合data:URL格式,格式如下:

url = canvas . toDataURL( [ type, ... ])

        规范规定,在未指定返回图片类型时,返回的图片格式必须为PNG格式,如果canvas没有任何像素,则返回值为:“data:,”,这是最短的data:URL,在text/plain资源中表现为空字符串。type的可以在image/png,image/jpeg,image/svg+xml等 MIME类型中选择。如果是image/jpeg,可以有第二个参数,如果第二个参数的值在0-1之间,则表示JPEG的质量等级,否则使用浏览器内置默认质量等级。

        下面的代码可以从ID为codeex的canvas中取得绘制内容,并作为DataURL传递给img元素,并显示。

var canvas =  document.getElementById('codeex');

var url = canvas.toDataURL();

//idmyimg的图片元素

myimg.src = url;

2 二维绘图上下文

        当使用一个canvas元素的getContext(“2d”)方法时,返回的是CanvasRenderingContext2D对象,其内部表现为笛卡尔平面坐标,并且左上角坐标为(0,0),在本平面中往右则x坐标增加和往下方y坐标增加。每一个canvas元素仅有一个上下文对象。其接口如下:

interface  CanvasRenderingContext2D {

// back-reference to the canvas

    readonly attribute HTMLCanvasElement  canvas;

// state

    void restore(); // pop state stack and  restore state

    void save(); // push state on state stack

// transformations (default transform is the identity matrix)

    void rotate(in float angle);

    void scale(in float x, in float y);

    void setTransform(in float m11, in float  m12, in float m21, in float m22, in float dx, in float dy);

    void transform(in float m11, in float  m12, in float m21, in float m22, in float dx, in float dy);

    void translate(in float x, in float y);

// compositing

    attribute float globalAlpha; // (default  1.0)

    attribute DOMString  globalCompositeOperation; // (default source-over)

// colors and styles

attribute  any fillStyle; // (default black)

    attribute any strokeStyle; // (default  black)

    CanvasGradient createLinearGradient(in float  x0, in float y0, in float x1, in float y1);

    CanvasGradient createRadialGradient(in  float x0, in float y0, in float r0, in float x1, in float y1, in float r1);

    CanvasPattern createPattern(in  HTMLImageElement image, in DOMString repetition);

    CanvasPattern  createPattern(in HTMLCanvasElement image, in DOMString repetition);

    CanvasPattern createPattern(in  HTMLVideoElement image, in DOMString repetition);

// line styles

    attribute DOMString lineCap; //  "butt", "round", "square" (default  "butt")

    attribute DOMString lineJoin; //  "miter", "round", "bevel" (default  "miter")

    attribute float lineWidth; // (default 1)

    attribute float miterLimit; // (default  10)

// shadows

    attribute float shadowBlur; // (default  0)

    attribute DOMString shadowColor; //  (default transparent black)

    attribute float shadowOffsetX; //  (default 0)

    attribute float shadowOffsetY; //  (default 0)

// rects

    void clearRect(in float x, in float y, in  float w, in float h);

    void fillRect(in float x, in float y, in  float w, in float h);

    void strokeRect(in float x, in float y,  in float w, in float h);

// Complex shapes (paths) API

    void arc(in float x, in float y, in float  radius, in float startAngle,      in  float endAngle, in boolean anticlockwise);

    void arcTo(in float x1, in float y1, in  float x2, in float y2, in float radius);

    void beginPath();

    void bezierCurveTo(in float cp1x, in  float cp1y, in float cp2x, in float cp2y, in float x, in float y);

    void clip();

    void closePath();

    void fill();

    void lineTo(in float x, in float y);

    void moveTo(in float x, in float y);

    void quadraticCurveTo(in float cpx, in  float cpy, in float x, infloat y);

    void rect(in float x, in float y, in float  w, in float h);

    void stroke();

    boolean isPointInPath(in float x, in  float y);

// text

    attribute DOMString font; // (default  10px sans-serif)

    attribute DOMString textAlign; //  "start", "end", "left", "right",  "center" (default: "start")

    attribute DOMString textBaseline; //  "top", "hanging", "middle",  "alphabetic", "ideographic", "bottom" (default:  "alphabetic")

    void fillText(in DOMString text, in float  x, in float y, optional in    float  maxWidth);

    TextMetrics measureText(in DOMString  text);

    void strokeText(in DOMString text, in  float x, in float y, optional in float maxWidth);

// drawing images

    void drawImage(in HTMLImageElement image,  in float dx, in float dy, optional in float dw, in float dh);

    void drawImage(in HTMLImageElement image,  in float sx, in float sy, in float sw, in float sh, in float dx, in float dy,  in float dw, in float dh);

    void drawImage(in HTMLCanvasElement  image, in float dx, in float dy, optional in float dw, in float dh);

    void drawImage(in HTMLCanvasElement  image, in float sx, in float sy, in float sw, in float sh, in float dx, in  float dy, in float dw, in float dh);

    void drawImage(in HTMLVideoElement image,  in float dx, in float dy, optional in float dw, in float dh);

    void drawImage(in HTMLVideoElement image,  in float sx, in float sy, in float sw, in float sh, in float dx, in float dy,  in float dw, in float dh);

// pixel manipulation

    ImageData createImageData(in float sw, in  float sh);

    ImageData createImageData(in ImageData  imagedata);

    ImageData getImageData(in float sx, in  float sy, in float sw, in float sh);

    void putImageData(in ImageData imagedata,  in float dx, in float dy, optional in float dirtyX, in float dirtyY, in float  dirtyWidth, in float dirtyHeight);

};

 

interface CanvasGradient {

// opaque object

    void addColorStop(in float offset, in  DOMString color);

};


interface CanvasPattern {

// opaque object

};


interface TextMetrics {

    readonly attribute float width;

};


interface ImageData {

    readonly attribute CanvasPixelArray data;

    readonly attribute unsigned long height;

    readonly attribute unsigned long width;

};


interface CanvasPixelArray {

    readonly attribute unsigned long length;

    getter octet (in unsigned long index);

    setter void (in unsigned long index, in  octet value);

};


2.1 canvas的状态

    每个上下文都包含一个绘图状态的堆,绘图状态包含下列内容:

    1、当前的transformation matrix.

    2、当前的clipping region.

    3、当前的属性值:fillStyle, font, globalAlpha, globalCompositeOperation, lineCap, lineJoin, lineWidth, miterLimit, shadowBlur, shadowColor, shadowOffsetX, shadowOffsetY, strokeStyle, textAlign, textBaseline

        注:当前path和当前bitmap不是绘图状态的一部分,当前path是持久存在的,仅能被beginPath()复位,当前bitmap是canvas的属性,而非绘图上下文。

context . restore() //弹出堆最上面保存的绘图状态

context . save() //在绘图状态堆上保存当前绘图状态

        绘图状态可以看作当前画面应用的所有样式和变形的一个快照。而状态的应用则可以避免绘图代码的过度膨胀。

2.1.1 context . restore()


2.1.2 context . save()


2.2 转换(Transformations)

        当建立形状和路径时,转换矩阵被应用到其坐标上。转换的执行顺序是严格按顺序的(注:原文是反向,经试验应该是按调用顺序的)。

        在做转换/变形之前先保存状态是一个良好的习惯。大多数情况下,调用restore 方法比手动恢复原先的状态要简单得多。

2.2.1 context . rotate(angle)

context . rotate(angle) //按给定的弧度旋转,按顺时针旋转

rotate方法旋转的中心始终是canvas的原点,如果要改变它,需要使用translate方法。


2.2.2 context .scale(x, y)

        context . scale(x, y) //按给定的缩放倍率缩放,1.0,为不变,参数比1.0小表示缩小,否则表示放大。默认情况下,canvas的1 单位就是1 个像素。举例说,如果我们设置缩放因子是0.5,1个单位就变成对应0.5 个像素,这样绘制出来的形状就会是原先的一半。同理,设置为2.0 时,1个单位就对应变成了2 像素,绘制的结果就是图形放大了2倍。


2.2.3 context . setTransform(m11, m12, m21, m22, dx, dy)

context . setTransform(m11, m12, m21, m22, dx, dy) //重设当前的转换


2.2.4 context . transform(m11, m12, m21, m22, dx, dy)

context . transform(m11, m12, m21, m22, dx, dy)

//矩阵变换,结果等于当前的变形矩阵乘上

m11 m21 dx

m12 m22 dy

0      0       1

后的结果


2.2.5 context . translate(x, y)

context . translate(x, y) //可以理解为偏移,向x,y方向偏移指定的量,其用来移动Canvas的原点到一个指定的值

2.2.5.1 示例

        下面是一个利用translate方法进行绘制螺旋图案的例子:

//绘制螺旋图案的函数

function drawSpirograph(ctx,R,r,O){

    var x1 = R-O;

    var y1 = 0;

    var i = 1;

    ctx.beginPath();

    ctx.moveTo(x1,y1);

    do {

        if (i>20000) break;

        var x2 = (R+r)*Math.cos(i*Math.PI/72)  - (r+O)*Math.cos(((R+r)/r)*(i*Math.PI/72))

        var y2 = (R+r)*Math.sin(i*Math.PI/72)  - (r+O)*Math.sin(((R+r)/r)*(i*Math.PI/72))

        ctx.lineTo(x2,y2);

        x1 = x2;

        y1 = y2;

        i++;

    } while (x2 != R-O && y2 != 0 );

    ctx.stroke();

}


//调用部分代码

context.fillRect(0,0,300,300);

for (var i=0;i<3;i++) {

    for (var j=0;j<3;j++) {

        context.save();

        context.strokeStyle =  "#9CFF00";

        context.translate(50+j*100,50+i*100);

        drawSpirograph(context,20*(j+2)/(j+1),-8*(i+3)/(i+1),10);

        context.restore();

    }

}


2.3 合成(Compositing)

2.3.1 context . globalAlpha

context . globalAlpha [ = value ] //0-1.0之间的数据,设定图像的透明度


2.3.2 context . globalCompositeOperation

context . globalCompositeOperation [ = value ]

//设定重叠图像的覆盖方式,可以设定为(注,值大小写敏感):

注意:下面所有图例中,B(蓝色方块)是先绘制的,即“已有的canvas 内容”,A(红色圆形)是后面绘制,即“新图形”。


2.4  颜色和风格

2.4.1 context. fillStyle

context . fillStyle [ = value ] //返回填充形状的当前风格,能被设置以用来改变当前的填充风格,其值可以是CSS颜色字串,也可以是CanvasGradient或者CanvasPattern对象,非法的值将被忽略。


2.4.2 context. strokeStyle

context . strokeStyle [ = value ] //返回当前描绘形状的风格,能被设置,其值同上。

        设置Javascript例子如下:

context.strokeStyle="#99cc33";

context.fillStyle='rgba(50,0,0,0.7)';

context.lineWidth=10;

context.fillRect(20,20,100,100);

context.strokeRect(20,20,100,100);


        绘制的图形如下所示。


2.4.3 gradient . addColorStop(offset, color)

gradient . addColorStop(offset, color) //在给定偏移的地方增加一个渐变颜色点,偏移量取值范围为0-1.0之间,否则产生一个INDEX_SIZE_ERR的异常,color为DOM字符串,如果不能解析,则抛出一个SYNTAX_ERR的异常。

2.4.4 context . createLinearGradient(x0, y0, x1, y1)

gradient = context . createLinearGradient(x0, y0, x1, y1) //建立一个线性渐变,如果任何一个参数不是有限值,则抛出一个NOT_SUPPORTED_ERR的异常。

2.4.4.1 addColorStop()

    定义和用法

        addColorStop()方法规定 gradient 对象中的颜色和位置。

        addColorStop()方法与 createLinearGradient()createRadialGradient() 一起使用。

        注释:您可以多次调用addColorStop() 方法来改变渐变。如果您不对 gradient 对象使用该方法,那么渐变将不可见。为了获得可见的渐变,您需要创建至少一个色标。

    JavaScript语法:    

gradient.addColorStop(stopcolor);

    参数值

        参数描述

        stop介于 0.0 与 1.0 之间的值,表示渐变中开始与结束之间的位置。

        color在结束位置显示的 CSS 颜色值

2.4.4.2 示例1

        设置Javascript例子如下:

var  gradient = context.createLinearGradient(0, 2, 420, 2);

gradient.addColorStop(0, 'rgba(200, 0, 0, 0.8)');

gradient.addColorStop(0.5, 'rgba(0, 200, 0, 0.7)');

gradient.addColorStop(1, 'rgba(200, 0, 200, 0.9)');

context.strokeStyle = "#99cc33";

context.fillStyle= gradient; //copyright codeex.cn

context.lineWidth = 10;

context.fillRect(20, 20, 400, 100);

context.strokeRect(20, 20, 400, 100);


        绘制的图形如下所示。

2.4.4.3 示例——通过多个addColorStop() 方法来定义渐变

        通过多个 addColorStop() 方法来定义渐变:

var c=document.getElementById("myCanvas");

var ctx=c.getContext("2d");

var grd=ctx.createLinearGradient(0,0,170,0);

grd.addColorStop(0,"black");

grd.addColorStop("0.3","magenta");

grd.addColorStop("0.5","blue");

grd.addColorStop("0.6","green");

grd.addColorStop("0.8","yellow");

grd.addColorStop(1,"red");


ctx.fillStyle=grd;

ctx.fillRect(20,20,150,100);


2.4.5 context. createRadialGradient(x0, y0, r0, x1, y1, r1)

gradient = context . createRadialGradient(x0, y0, r0, x1, y1, r1) //建立一个径向渐变,如果任何一个参数不是有限值,则抛出一个NOT_SUPPORTED_ERR的异常。假如r0或r1为负值,则抛出INDEX_SIZE_ERR的异常。

2.4.5.1 绘制过程

        createRadialGradient(x0,y0,r0,x1,y1,r1)方法有六个参数,前三个参数表示开始的圆,其圆点在(x0,y0),半径为r0,后三个表示结束的圆,参数意义同上。其绘制过程如下:

    1. 如果起始圆和结束圆重叠,则不绘制任何东西,并终止步骤;

    2. x(w) = (x1-x0)w + x0 y(w) = (y1-y0)w + y0 r(w) =(r1-r0)w + r0在以(x(w),y(w))为圆点,r(w)为半径的圆周上所有点的颜色均为Color(w)。

    3. 对于任意的w取值(-∞ -- +∞),确保r(w)>0,总是可以知道画布中已知点的颜色。

        言而总之:这个效果就是建立一个圆锥体(手电筒效果)渲染效果,圆锥体的开始圆使用开始颜色偏移量为0,圆锥体的结束圆使用颜色偏移量为1.0,面积外的颜色均使用透明黑。


2.4.5.2 示例

        设置Javascript例子如下:

var gradient  = context.createRadialGradient (100,100, 20, 300,300,80);

gradient.addColorStop(0,'rgba(200,0,0,0.8)');

gradient.addColorStop(1,'rgba(200,0,200,0.9)');

context.strokeStyle="#99cc33";

context.fillStyle= gradient;    //'rgba(50,0,0,0.7)';

context.lineWidth=10;

context.fillRect(10,10,400,400);

context.strokeRect(10,10,400,400);


绘制的图形如下所示。

2.4.6 context . createPattern(image, repetition)

        上面提到可以作为渲染风格还有图案对象:CanvasPattern,其调用格式如下:

pattern = context . createPattern(image, repetition)

        本方法用指定的图像和重复方向建立一个画布图案对象,image参数可以为img,canvas,video元素中的任一个,如果不满足此条件,则抛出TYPE_MISMATCH_ERR异常,如果图片编码未知或没有图像数据,则抛出INVALID_STATE_ERR异常;第二个参数可以是下列值:

    repeat    默认参数,如果为空,则为此参数,表示两个方向重复

    repeat-x    仅水平重复

    repeat-y    仅垂直重复

    no-repeat    不重复

        如果image参数是一个HTMLImageElement对象,但对象的complete属性是false,则执行时抛出INVLAID_STATE_ERR异常;

        如果image参数是一个HTMLVideoElement对象,但其readyState属性是HAVE_NOTHING或HAVE_METADATA,则执行时抛出抛出INVLAID_STATE_ERR异常;

        如果image参数是一个HTMLCanvasElement对象,但其width属性或height属性是0,则执行时抛出抛出INVLAID_STATE_ERR异常。

        图案的绘制时从左上角开始的,根据不同的参数进行重复绘制。如果传递的图片是动画,则选取海报或第一帧作为其绘制图案源,如果使用HTMLVideoElement为对象,则当前播放位置帧被作为图案源。


2.4.6.1 示例

        设置HTML的核心代码如下:

        设置Javascript例子如下:

var imgSrc = document.getElementById('psrc')

var pattern =  context.createPattern(imgSrc,'repeat');

context.strokeStyle="#99cc33";

context.fillStyle= pattern;//by codeex.cn

context.lineWidth=10;

context.fillRect(10,10,200,220);

context.strokeRect(10,10,200,220);

        在IE9中的显示效果如图所示:


2.5 线风格

2.5.1 context . lineCap

        操作线风格的方法有4个,格式如下:

context . lineCap [ = value ] //返回或设置线段的箭头样式,仅有三个选项:butt(默认值),round,square;其他值忽略

2.5.2 context . lineJoin

context . lineJoin [ = value ] ///返回或设置线段的连接方式,仅有三个选项:miter(默认值),round,bevel;其他值忽略


2.5.3 context . lineWidth

context . lineWidth [ = value ] //返回或设置线段的线宽,非大于0的值被忽略;默认值为1.0;


2.5.4 context . miterLimit

context . miterLimit [ = value ] //返回或设置线段的连接处的斜率,非大于0的值被忽略;默认值为10.0。本属性翻译不够准确,请参看英文部分


2.5.5 线宽

        线宽是指给定路径的中心到两边的粗细。换句话说就是在路径的两边各绘制线宽的一半。因为画布的坐标并不和像素直接对应,当需要获得精确的水平或垂直线的时候要特别注意。

        想要获得精确的线条,必须对线条是如何描绘出来的有所理解。见下图,用网格来代表canvas 的坐标格,每一格对应屏幕上一个像素点。在第一个图中,填充了 (2,1) 至 (5,5) 的矩形,整个区域的边界刚好落在像素边缘上,这样就可以得到的矩形有着清晰的边缘。

        如果你想要绘制一条从 (3,1) 到 (3,5),宽度是 1.0的线条,你会得到像第二幅图一样的结果。实际填充区域(深蓝色部分)仅仅延伸至路径两旁各一半像素。而这半个像素又会以近似的方式进行渲染,这意味着那些像素只是部分着色,结果就是以实际笔触颜色一半色调的颜色来填充整个区域(浅蓝和深蓝的部分)。

        要解决这个问题,你必须对路径施以更加精确的控制。已知粗 1.0 的线条会在路径两边各延伸半像素,那么像第三幅图那样绘制从(3.5,1) 到 (3.5,5) 的线条,其边缘正好落在像素边界,填充出来就是准确的宽为1.0 的线条。

        对于那些宽度为偶数的线条,每一边的像素数都是整数,那么你想要其路径是落在像素点之间(如那从(3,1) 到 (3,5)) 而不是在像素点的中间。如果不是的话,端点上同样会出现半渲染的像素点。

2.6 阴影(Shadows)

        有关阴影的四个全局属性将影响所有的绘画操作。有关定义如下:

context . shadowBlur [ = value ] //返回或设置阴影模糊等级,非大于等于0的值被忽略;

context . shadowColor [ = value ] //返回或设置阴影颜色

context . shadowOffsetX [ = value ]

context . shadowOffsetY [ = value ] //返回或设置阴影的偏移量

        注意:上面的值均不受坐标转换的影响,可以看做是绝对值。

        在上面的例子中增加下列语句,可以得到设置阴影的图像:

context.shadowBlur=7;

context.shadowColor='rgb(200,0,0)';

context.shadowOffsetX = 3;

context.shadowOffsetY=3;


2.7 简单形状(矩形)

        形状的绘制不影响当前路径(path),形状是剪切区域的主题,也是阴影(Shadow)效果,全局透明(alpha),全局组合(composition)操作等的主题。其由下面三个方法来进行简单的操作:

2.7.1 context. clearRect(x, y, w, h)

context. clearRect(x, y, w, h) \\在给定的矩形内清除所有的像素为透明黑(transparentblack)

2.7.2 context. fillRect(x, y, w, h)

context. fillRect(x, y, w, h) //用当前的填充风格填充给定的区域

2.7.3 context . strokeRect(x, y, w, h)

context. strokeRect(x, y, w, h) //使用当前给定的线风格,绘制一个盒子区域,影响其绘制风格的有:strokeStyle,lineWidth,lineJoin,miterLimit(可能)。

2.8 复杂形状(路径-paths)

        绘图上下文总有一个当前路径,并且是仅此一个,它不是绘图状态的一部分。

        一个路径有0个或多个子路径列表。每个子路径包含一个或多个点列表(这些点组成直的或弯曲的线段),和一个标识子路径是否闭合的标志。少于两个点的子路径在绘图时被忽略。操作这些形状的方法稍微多些,如下所示:

        默认情况下,图形上下文的路径有0个子路径。


2.8.1 路径起始函数

    调用格式:

context . beginPath() //清空子路径

context . closePath() //闭合路径


        方法概述:

2.8.1.1 context. beginPath()

beginPath方法重设绘图上下文的子路径列表,并清空所有的子路径。


2.8.1.2 context. closePath()

        ClosePath方法在绘图上下文如果没有子路径时,什么也不做;否则,它先把最后一个子路径标示为闭合,然后建立一个包含最后子路径的第一个点的子路径,并加入到绘图上下文。有点拗口,其一般可以看为,假如最后一个子路径,我们命名为spN,假设spN有多个点,则用直线连接spN的最后一个点和第一个点,然后关闭此路径和moveTo到第一个点。


2.8.2 绘制函数

        调用格式:

context . stroke()

context . fill()

context . clip()

        方法概述:


2.8.2.1 context. stroke()

        stroke方法使用lineWidth,lineCap,lineJoin,以及strokeStyle对所有的子路径进行填充。

2.8.2.2 context. fill()

        fill方法使用fillStyle方式填充子路径,未闭合的子路径在填充式按照闭合方式填充,但并不影响实际的子路径集合。

2.8.2.3 context. clip()

        clip方法使用计算所有的子路径而建立新的剪切区域,未闭合的子路径在填充式按照闭合方式填充,但并不影响实际的子路径集合,新的剪切区域将替换当前的剪切区域。

2.8.3 剪切(clip)

        裁切路径和普通的 canvas 图形差不多,不同的是它的作用是遮罩,用来隐藏没有遮罩的部分。如下图所示。红边五角星就是裁切路径,所有在路径以外的部分都不会在canvas 上绘制出来。

        如果和上面介绍的 globalCompositeOperation 属性作一比较,它可以实现与source-in 和 source-atop 差不多的效果。最重要的区别是裁切路径不会在canvas 上绘制东西,而且它永远不受新图形的影响。这些特性使得它在特定区域里绘制图形时相当好用。

2.8.3.1 随机星星示例

        效果图:

        首先,我画了一个与 canvas 一样大小的黑色方形作为背景,然后移动原点至中心点。然后用 clip方法创建一个弧形的裁切路径。裁切路径也属于 canvas 状态的一部分,可以被保存起来。如果我们在创建新裁切路径时想保留原来的裁切路径,我们需要做的就是保存一下canvas 的状态。

        裁切路径创建之后所有出现在它里面的东西才会画出来。在画线性渐变时这个就更加明显了。然后在随机位置绘制50 大小不一(经过缩放)的颗,当然也只有在裁切路径里面的星星才会绘制出来。

        代码如下:

function draw()  {

    var ctx = document.getElementById('canvas').getContext('2d');

    ctx.fillRect(0,0,150,150);

    ctx.translate(75,75);

// Create a  circular clipping path

    ctx.beginPath();

    ctx.arc(0,0,60,0,Math.PI*2,true);

    ctx.clip();

// draw  background

    var lingrad = ctx.createLinearGradient(0,-75,0,75);

    lingrad.addColorStop(0, '#232256');

    lingrad.addColorStop(1, '#143778');

    ctx.fillStyle = lingrad;

    ctx.fillRect(-75,-75,150,150);

// draw stars

    for (var j=1;j<50;j++){

        ctx.save();

        ctx.fillStyle = '#fff';

        ctx.translate(75-Math.floor(Math.random()*150),  75-Math.floor (Math. random () *150));

        drawStar(ctx,Math.floor(Math.random()*4)+2);

        ctx.restore();

    }

}


function  drawStar(ctx,r){

    ctx.save();

    ctx.beginPath()

    ctx.moveTo(r,0);

    for (var i=0;i<9;i++){

        ctx.rotate(Math.PI/5);

        if(i%2 == 0) {

            ctx.lineTo((r/0.525731)*0.200811,0);

        } else {

            ctx.lineTo(r,0);

        }

    }

    ctx.closePath();

    ctx.fill();

    ctx.restore();

}

2.8.4 辅助方法—context. isPointInPath(x, y)

        调用格式:

context . isPointInPath(x, y)


        方法概述:

        给定的坐标(x,y)是否在当前路径中,坐标(x,y)为绘图坐标系坐标,并不受转换的影响。

2.8.5 moveTo方法

    调用格式:

context . moveTo(x, y)

    方法概述:

        建立新的子路径,并制定其第一个点为(x,y)。

2.8.6 lineTo方法

    调用格式:

context . lineTo(x, y)

    方法概述:

        如果绘图上下文没有子路径,则其等同于moveTo(x,y),否则,其建立一条在子路径最后一个点到给定点的直线,并增加(x,y)到子路径中。

2.8.7 rect方法

    调用格式:

context . rect(x, y, w, h)

    方法概述:

        本方法建立二个新的子路径,第一个子路径包含四个点:(x,y),(x+w,y),(x+w,y+h),(x,y+h),四个点的连接方式为直线,该子路径被标示为闭合路径;最后再增加一个子路径,其仅有一个点(x,y)。

2.8.8 圆弧context. arc(x, y, radius, startAngle, endAngle, anticlockwise)

    方法调用格式:

context . arc(x, y, radius, startAngle,endAngle, anticlockwise)

    方法概述:

        本方法先增加一条直线到子路径列表,然后增加一个圆弧子路径到子路径列表。直线子路径是由上一个点到圆弧子路径的起始点,而圆弧则为按照给定的开始角度、结束角度和半径描述的按照给定的方向[布尔类型,anticlockwise-逆时针(true)]圆弧上;假如半径为负值,抛出INDEX_SIZE_ERR的异常;

    JS代码:

context.beginPath();

context.moveTo(100,50);

context.arc(250,50,50,1.5708,3.14,true);

context.stroke();

注释掉moveTo语句,则仅仅绘制圆弧:

2.8.9 最短圆弧context. arcTo(x1, y1, x2, y2, radius)

    方法调用格式:

context . arcTo(x1, y1, x2, y2, radius)

    方法概述:

        本方法绘制出子路径最后一个点(x0,y0)和(x1,y1)以及(x1,y1)和(x2,y2)构成的两条直线间半径为radius的最短弧线,并用直线连接(x0,y0);假如半径为负值,抛出INDEX_SIZE_ERR的异常;

        如图所示,绘制曲线由1,开始绘制。

        JS代码如下:

context.beginPath();

context.moveTo(150,50);

context.arcTo(200,50,100,200,20);

context.arcTo(200,50,100,200,40);

context.arcTo(200,50,100,200,80);

context.arcTo(200,50,100,200,120);

context.arcTo(200,50,100,200,160);

context.stroke();

2.8.10 二次方、三次方贝塞尔曲线

        贝塞尔曲线的一般概念:在数学的数值分析领域中,贝赛尔曲线(Bezier curve)是电脑图形学中相当重要的参数曲线。更高维度的贝赛尔曲线被称作贝塞尔曲面。对于n阶贝塞尔曲线可如下推断,给定P0、P1、P2…Pn,其贝赛尔曲线即为

        用平常话说,n阶的贝赛尔曲线就是双n-1阶贝赛尔曲线之间的插值。

        由公式可以得出二次方贝塞尔曲线公式如下:

        TrueType字型就运用了以贝塞尔样条组成的二次方贝赛尔曲线。


    方法调用格式:

bezierCurveTo(cp1x, cp1y, cp2x, cp2y, x, y)

quadraticCurveTo(cpx, cpy, x, y)

    方法概述:

        上面分别是三次贝赛尔曲线和二次贝赛尔曲线的调用格式。其主要区别在于控制曲线的控制点式不一样的。其起始点均为子路径的最后一个点,结束点均为(x,y);在最后均要把(x,y)点加入到子路径中。

        其绘制图形的例子如下,三次贝赛尔曲线有两个红点作为曲线平滑的控制点,而二次贝塞尔曲线仅有一个控制点。

2.8.10.1 二次贝赛尔曲线quadraticCurveTo(cpx, cpy, x,y)


2.8.10.2 三次贝赛尔曲线bezierCurveTo(cp1x, cp1y, cp2x,cp2y, x, y)


2.9 文字

        绘图上下文提供了得到和设置文字样式的接口方法。

2.9.1 context.font[=value]

        获得和设置文字设置: context.font[=value],可以参考CSS中对font风格的设置。

2.9.2 context.textAlign[=value]

        获取或设置文字对齐方式:context.textAlign[=value],取值如下:

start                默认值,与canvas风格中的direction定义有关

end                 与canvas风格中的direction定义有关

left                  左

right                右

center             居中


2.9.3 context.textBaseline

        获得和设置文字对齐基线:context.textBaseline[=value],value的取值如下:

2.9.4 context.fillText(text, x, y[,maxWidth])

        绘制填充的文字,

2.9.5 context.strokeText(text, x, y[,maxWidth])

        对文字进行描边,不填充内部区域。

2.9.6 context.measureText(text)

        按照当前字体对给定的文字进行测量:

metrics = context.measureText(text),该方法返回一个TextMetrics对象,可以调用对象的width属性得到文字的宽度。

2.10 绘制图片

2.10.1 drawImage方法

        要在绘图上下文中绘制图片,可以使用drawImage方法。该方法有三种不同的参数:

        其中的image参数可以是HTMLImageElement、HTMLCanvasElement和HTMLVideoElement中的任一个对象。

        绘制参数的含义可以参看下图:

    异常:

        如果第一个参数不是指定的元素类型,抛出一个TYPE_MISMATCH_ERR异常,如果图片不能进行解码,则抛出INVALID_STATE_ERR异常,如果第二个参数不是允许的值,则抛出SYNTAX_ERR异常。    

    参数默认值:

        如果没有指定dw和dh,则默认等于sw和sh,如果sx,sy,sw,sh均没有提供,则默认为sx,xy=0,0;sw和sh为图片的像素宽高。

2.10.2 图片调用方式

        下面给出图片的几种调用方式:

    1.引用页面内图片:

        我们可以通过 document.images 集合、document.getElementsByTagName方法又或者document.getElementById 方法来获取页面内的图片(如果已知图片元素的 ID。

    2.引用canvas元素

        用 document.getElementsByTagName 或document.getElementById 方法来获取其它 canvas 元素。但你引入的应该是已经准备好的canvas。一个常用的应用就是为另一个大的 canvas 做缩略图。

    3.创建图像

        我们可以用脚本创建一个新的 Image对象,但这种方法的主要缺点是如果不希望脚本因为等待图片装置而暂停,还得需要突破预装载。

var img = new Image(); // Create new Image

img.src = 'myImage.png'; // Set source path

        当脚本执行后,图片开始装载。若调用 drawImage 时,图片没装载完,脚本会等待直至装载完毕。如果不希望这样,可以使用onload 事件:

var img = new Image(); // Create new Image

img.onload = function(){

    // 执行drawImage 语句

}

img.src = 'myImage.png'; // Set source path

        如果你只用到一张图片的话,这已经够了。但一旦需要不止一张图片,那就需要更加复杂的处理方法,但图片预装载策略超出本规范的范围。

    4.通过 data: url 方式嵌入图像

        我们还可以通过 data: url 方式来引用图像。Data urls 允许用一串Base64 编码的字符串的方式来定义一个图片。其优点就是图片内容即时可用,无须再到服务器兜一圈。(还有一个优点是,可以将 CSS,JavaScript,HTML 和 图片全部封装在一起,迁移起来十分方便。)缺点就是图像没法缓存,图片大的话内嵌的url数据会相当的长,例如:

var img_src = '';

        有兴趣的朋友可以使用<img src=’上述变量值’>的方法显示出上面的图片。

2.11 像素级操作(good)

        2D Context API 提供了三个方法用于像素级操作:createImageData, getImageData, 和putImageData。

        ImageData对象保存了图像像素值。每个对象有三个属性: width, height 和data。data 属性类型为CanvasPixelArray,用于储存width*height*4个像素值。每一个像素有RGB值和透明度alpha值(其值为 0 至255,包括alpha在内)。像素的顺序从左至右,从上到下,按行存储。

        Canvas提供像素级数据,为很多算法的应用提供了平台,实现photoshop中的众多神奇图像效果在前端都已成为可能。甚至于我们还能实现电影的蓝幕效果——针对video的帧,通过drawImage绘进canvas,再做rgba处理,将规定颜色的像素的alpha值设为0,就能使特定部分变成透明,进而实现视频合成。查看演示

canvas的像素级操作——1.引子

http://cssass.com/blog/2012/1140.html

canvas的像素级操作——2.RGBA通道调色

http://cssass.com/blog/2012/1158.html

canvas的像素级操作——3.使用卷积矩阵

http://cssass.com/blog/2012/1165.html

canvas的像素级操作——4.关注性能

http://cssass.com/blog/2012/1179.html

[Color]彩色转灰度算法彻底学习

http://www.cnblogs.com/zyl910/archive/2006/05/22/2186658.html

在HTML5 的CANVAS 中应用卷积矩阵对图像处理

http://shawphy.com/2011/08/convolution-matrix-in-canvas.html

2.11.1 createImageData方法

imagedata = context . createImageData(sw, sh)

imagedata = context . createImageData(imagedata)

    方法概述:

        createImageData方法根据给定的CSS像素宽高或指定的imagedata具有的宽高建立一个ImageData对象,该对象为透明黑。该方法具体实例化一个新的空ImageData对象。

2.11.2 getImageData方法

imagedata = context . getImageData(sx,sy,sw, sh)

    方法概述:

        getImageData方法根据给定的绘图画布矩形面积(sx,sy,sw,sh),生成画布上该矩形面积的图形内容,并综合为ImageData对象返回。画布外的像素作为透明黑返回。

2.11.3 putImageData方法

imagedata = context . putImageData(imagedata, dx, dy [, dirtyX, dirtyY, dirtyWidth, dirtyHeight ])

    方法概述:

        在绘图画布上绘制给定的ImageData对象。假如脏矩形被提供,则只有在脏矩形上面的像素被绘制。本方法对全局透明、阴影和全局组合属性均忽略。

        异常:假如第一个参数不是ImageData对象,抛出TYPE_MISMATCH_ERR异常,假如任一数字参数是无穷或非数字,则抛出NOT_SUPPORTED_ERR错误。

        putImageData参数有(ImageData,dx, dy [, DirtyX] [, DirtyX] [, DirtyWidth] [, DirtyHeight])

    imageData:包含了图像的width,height,还有一个CanvasPixelArray,前面已经提了。

    dx, dy:表示绘图起始位置。相对于canvas区域左上角。

        后面四个可选参数:表示可见区范围。相对于起绘点,即上面的参数dx,dy表示的点。缺省为0,0,ImageData.width,ImageData.height。

2.11.3.1 具体用法

        通过将源canvas中像素数据ImageData,输出(putImageData)到新的canvas中,达到复制作用。

<script type="text/javascript">

function draw(){

            /*在canvas中绘制image*/

            var canvas1 =document.getElementById("MyCanvas");

            var ctx1 =canvas1.getContext("2d");

                        ctx1.drawImage(imgObj,0,0);

}

function clone(){

            /*在canvas2中绘制canvas1——普通复制*/

            var origin =document.getElementById("MyCanvas");

            var canvas2 =document.getElementById("YourCanvas");

            var ctx2 =canvas2.getContext("2d");

            ctx2.drawImage(origin,0,0);//drawImage不仅可以绘image,也可以绘canvas对象,甚至还可以绘video的帧

}

function cloneData(canvasObj){

            /*获取canvas1中的ImageData,在canvas3中输出 ——像素级复制*/

            var origin =document.getElementById("MyCanvas");

            var canvas3 =document.getElementById("GodCanvas");

            var ctx3 =canvas3.getContext("2d");

            var canvasCtx = origin.getContext("2d");

            var imagePix = canvasCtx.getImageData(0,0,origin.width,origin.height); //获取canvas1绘图环境下的参数范围内的imageData。

            ctx3.putImageData(imagePix,0,0);    //putImageData输出图像

}

/* 以下与示例代码无关*/

function Load(canvas){

            /* canvas Loading效果*/

            var backCtx =canvas.getContext('2d');

                        backWidth= canvas.width;

                        backHeight= canvas.height;

            var drawIntervalID,

                        spokes= 7;

            var      drawPad =document.createElement('canvas');

                        drawPad.width= 30;

                        drawPad.height= 30;

            var      drawCtx = drawPad.getContext('2d');

                        drawCtx.translate(drawPad.width/2,drawPad.height/2);

                        drawCtx.lineWidth= 5;

                        drawCtx.lineCap= "round";

                        drawCtx.strokeStyle= "rgba(0,0,0,0.1)";

                        drawCtx.fillStyle= "#fff";

            var draw =function(){

                        drawCtx.fillRect(0,0,drawPad.width ,drawPad.height);

                        drawCtx.rotate(Math.PI*2/spokes);

                        for(var i=0; i

                                    drawCtx.rotate(Math.PI*2/spokes);

                                    drawCtx.beginPath();

                                    drawCtx.moveTo(0,8);

                                    drawCtx.lineTo(0,10);

                                    drawCtx.stroke();

                        }

                        backCtx.drawImage(drawPad,(backWidth- drawPad.width)/2, (backHeight - drawPad.height)/2);

            }

            this.loading =function(){

                         drawIntervalID = setInterval(draw,200);

            }

            this.loaded =function(){

                        clearInterval(drawIntervalID);

                        backCtx.clearRect((backWidth- drawPad.width)/2, (backHeight - drawPad.height)/2, drawPad.width,drawPad.height);

            }

}

var imgObj = document.getElementById("imgObj");

var canvas = document.getElementsByTagName('canvas');

for(var i = 0, l = canvas.length; i < l; i++ ){

            var loadObj = newLoad(canvas[i]);

            (function(obj){

                        obj.loading();

                        imgObj.addEventListener('load',obj.loaded,false);

            })(loadObj);

}

imgObj.addEventListener('load',function(){

            draw();

            clone();

            cloneData();

},false);

</script>

2.11.4 示例

        下面展示了对一张图片进行反色、透明的一个例子,从例子中可以看出,有了像素级的控制能力,我们可以很轻易的对原有图片进行各种图像滤镜操作。图示如下:

        JS代码:[当你在word中选择上面的图片,会发现反色滤镜常用在选择操作里]

var imgSrc =  document.getElementById('codeex.cn')

context.drawImage(imgSrc,10,10);

var imgd =  context.getImageData(10,10,100,122);

var pix =  imgd.data;


//反色处理

for(var  i=0,n=pix.length;i

{

    pix[i] = 255 - pix[i]; //红

    pix[i+1] = 255-pix[i+1]; //绿

    pix[i+2] = 255-pix[i+2]; //蓝

    pix[i+3] = pix[i+3]; //alpha

}


context.putImageData(imgd,130,10);

imgd =  context.getImageData(10,10,100,122);

pix =  imgd.data;

//透明处理 透明度0.6

for(var  i=0,n=pix.length;i

{

    pix[i] = pix[i]; //红

    pix[i+1] = pix[i+1]; //绿

    pix[i+2] = pix[i+2]; //蓝

    pix[i+3] = pix[i+3]*0.6; //alpha

}

context.putImageData(imgd,260,10);

2.11.5 性能优化示例

canvas像素级操作

http://blog.sina.com.cn/s/blog_502364000100qwc8.html

2.12 绘图模型

        在本文描述的画布中绘图,浏览器一般按照下面的顺序进行绘制:

    1.准备形状或图片,此时图片假设为A,形状必须被所有属性描述的形状,且经过坐标转换;

    2.当绘制阴影时,准备图片A,并绘制阴影,形成图片B;

    3.当绘制阴影时,为B的每个像素乘上alpha值;

    4.当绘制阴影时,则根据组合参数对B和本画布剪贴区域内的图片进行组合;

    5.在图片A上每个像素乘上alpha值;

    6.在图片A上根据组合参数对A和本画布剪贴区域内的图片进行组合;

3 Canvas动画库——KineticJS

以下教程是根据2012年教程整理的,部分接口有调整,后续注意逐步整理更新

KineticJS中文系列教程

http://iysm.net/?paged=2

How It Works

https://github.com/ericdrowell/KineticJS/wiki

4 Canvas动画库——Collie

Collie——基于 HTML5 的高性能

JavaScript 动画库

http://www.cnblogs.com/lhb25/archive/2012/12/17/create-highly-optimized-animations-using-html5.html

5 Canvas动画库——EaselJS

EaselJS(CreateJS)屏幕适配

http://abellee.github.io/blog/easeljs/2014/07/11/EaselJS(CreateJS)%E5%B1%8F%E5%B9%95%E9%80%82%E9%85%8D.html

CreateJS基础

http://www.jikexueyuan.com/course/276.html


6 特效合集

6.1 星星图形

function drawStar(ctx,r){

    ctx.save();

    ctx.beginPath()

    ctx.moveTo(r,0);

    for (var i=0;i<9;i++){

        ctx.rotate(Math.PI/5);

        if(i%2 == 0) {

           ctx.lineTo((r/0.525731)*0.200811,0);

        } else {

            ctx.lineTo(r,0);

        }

    }

    ctx.closePath();

    ctx.fill();

    ctx.restore();

}


7 参考链接

HTML5 canvas addColorStop() 方法

http://www.w3school.com.cn/tags/canvas_addcolorstop.asp

canvas像素级操作

http://blog.sina.com.cn/s/blog_502364000100qwc8.html


专栏——HTML5 Canvas编程

http://blog.csdn.net/column/details/canvas-programming.html


专栏——html5 Canvas画图系列教程目录

http://jo2.org/html5-canvas-tutorial-list/


[Canvas系列]Canvas绘制圆弧形状_04

http://blog.csdn.net/baihuaxiu123/article/details/53619435


HTML5 Canvas 画圆教程

http://www.108js.com/article/article7/70206.html?id=1036


6.5 径向渐变

http://www.lvyestudy.com/css3/css3_6.5.aspx

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容

  • 一:canvas简介 1.1什么是canvas? ①:canvas是HTML5提供的一种新标签 ②:HTML5 ...
    GreenHand1阅读 4,679评论 2 32
  • 一、图形的组合方式 globalAlpha是一个介于0和1之间的值(包括0和1),用于指定所有绘制的透明度。默认值...
    空谷悠阅读 1,267评论 0 0
  • 一、canvas简介 1.1 什么是canvas?(了解) 是HTML5提供的一种新标签 Canvas是一个矩形区...
    J_L_L阅读 1,516评论 0 4
  • 啥是canvas? HTML5 标签用于绘制图像(通过脚本,通常是 JavaScript)。不过, 元素本身...
    kiaizi阅读 767评论 0 4
  • 1 CALayer IOS SDK详解之CALayer(一) http://doc.okbase.net/Hell...
    Kevin_Junbaozi阅读 5,148评论 3 23