龙空技术网

开源Html5画布操作类库Fabric.js进阶提高——第二部分(上)

三石科技观察 49

前言:

此时我们对“js透明度代码”可能比较注意,咱们都想要分析一些“js透明度代码”的相关文章。那么小编在网上网罗了一些关于“js透明度代码””的相关文章,希望大家能喜欢,同学们一起来了解一下吧!

在这个系列的第一部分,我们只是开始熟悉Fabric.js。我们研究了使用Fabric的原因、它的对象模型和对象层级,不同种类的可用实体--简单图形、图像和复杂路径。我们也学习了怎样在画布上使用Fabric对象进行简单的操作。

现在,这些基础部分可以抛一边了,咱们玩点有意思的。

动画

没有什么NB的画布类库不包括动画功能的,Fabric也不例外。既然有如此强大的对象模型和图形能力,那么如果没有内置的动画助手就有点太可惜了。

还记得改变任何对象的属性有多简单吗?我们只需要调用set方法,传递合适的值就可以了:

rect.set('angle', 45);

那么,驱动一个对象的动画效果也很容易,每一个Fabric对象都有animate方法,它可以使对象动起来;

rect.animate('angle', 45, {

onChange: canvas.renderAll.bind(canvas)

});

第一个参数是要动起来的属性,第二个参数是动画结束时的值。如果举行具有-15度的值,并且传递45给它,它将在-15度到45度之间动起来。第三个参数是可选的对象,指定动画更精细的细节--持续时间、回调、过度等。

动画的一个方便的特点是它也支持相对值,举个例子,如果你想把对象的left属性动起来100像素,你可以这样做:

rect.animate('left', '+=100', { onChange: canvas.renderAll.bind(canvas) });

类似的,逆时针旋转对象5度,可以像这样完成:

rect.animate('angle', '-=5', { onChange: canvas.renderAll.bind(canvas) });

你可能想知道为什么我们总是在那里指定“Onchange”回调?第三个参数是可选的吗?是的,但是在每一个动画帧后调用canvas.renderAll可以让我们看到真正的动画!你要明白,当我们调用animate方法时,它只是随着时间改变动画属性值,遵循特定的算法(比如过度)。因此rect.animate('angle', 45)将改变对象角度,但是在每次角度改变后都不会重绘画布屏幕。我们当然需要重绘来看到真正的动画。

记住在画布的表面下,具有完整的对象模型,对象具有它们自己的属性和关系,画布只负责向外界投射它们的存在。

动画不在每次改变后自动重绘画布的原因是执行效率的原因,毕竟,我们可能有成百上千个对象在画布上,如果每一个对象都试图重绘屏幕不是什么好事。在具有很多对象的情况下,你可以使用像requestAnimationFrame(或基于其他定时器的)循环来连续地绘制画布,而不是调用每个对象的renderAll方法。但是,大部分情况下,你可能需要显示指定canvas.renderAll作为"onChange"的回调。

那么那些选项是我们可以传递给动画的?

from:允许指定动画属性的开始值(如果我们不想使用当前值)。duration:默认500(毫秒),可以用来改变动画的持续时间。onComplete:动画结束时的回调。easing:过度函数。

所有这些选项都是显而易见的,也许除了那个easing,让我们走近看一看。

默认,动画使用“easeInSine”过度函数,如果这个不是你需要的,在fabric.util.ease下有一系列可用过度选项,比如,如果我们想用弹性风格移动对象到右面:

rect.animate('left', 500, {

onChange: canvas.renderAll.bind(canvas),

duration: 1000,

easing: fabric.util.ease.easeOutBounce

});

注意,fabric.util.ease.easeOutBounce是一个过度选项,其他值得注意的包括easeInCubic、easeOutCubic、easeInElastic、easeOutElastic、easeInBounce和easeOutExpo。

因此,这些包括Fabric的动画部分,只是想让你知道什么是可能的--你可以动态改变对象角度使它旋转,改变left/top属性使它移动,改变width/height使它收缩或增长,改变透明度使它淡入淡出等等。

图像滤镜

在这个系列的第一部分,我们学习了在Fabric里如何处理图像。有一个fabric.Image构造函数,可以接收图像元素,也有一个fabric.Image.fromURL方法,可以通过URL字符串创建图像实例,所有这些图像和其他对象一样,都可以在画布上被移动和重绘。

就像使用图像一样有趣的是,把图像滤镜应用与它们会更酷!

默认情况下,对于启没启用WEBGL的浏览器,Fabric都没提供几个滤镜,但是定义自己的滤镜也很简单。一些内置的滤镜你可能也很熟悉--去除白色背景滤镜、灰度滤镜、翻转或亮度滤镜,其他可能不太流行--颜色矩阵、老照片、降噪。

那么我们在Fabric里如何对图像应用滤镜呢?fabric.Image的每个实例都有"filters"属性,这是一个简单的滤镜数组,数组中的每一个对象都是一个Fabric滤镜实例或者你自定义的滤镜实例。

让我们创建一个灰度图像:

fabric.Image.fromURL('pug.jpg', function(img) {

// add filter

img.filters.push(new fabric.Image.filters.Grayscale());

// apply filters and re-render canvas when done

img.applyFilters();

// add image onto canvas (it also re-render the canvas)

canvas.add(img);

});

老照片版本的图像了解下?

fabric.Image.fromURL('pug.jpg', function(img) {

img.filters.push(new fabric.Image.filters.Sepia());

img.applyFilters();

// add image onto canvas (it also re-render the canvas)

canvas.add(img);

});

既然"filters" 属性是一个简单的数组,我们可以使用通常的方式来应用我们需要的操作--删除滤镜(通过pop、splice或shift),添加滤镜(通过push、splice、unshift),甚至合并多个滤镜。当我们调用applyFilters时,所有在"filters" 数组中的滤镜都会被一个接一个应用。我们试着创建一个应用了老照片和亮度滤镜的图像:

fabric.Image.fromURL('pug.jpg', function(img) {

img.filters.push(

new fabric.Image.filters.Sepia(),

new fabric.Image.filters.Brightness({ brightness: 100 }));

img.applyFilters();

canvas.add(img);

});

注意,我们也传递了{ brightness: 100 }参数给亮度滤镜,这是因为有些滤镜可以没有额外的配置(比如灰度、翻转、老照片),有些为了更好的控制表现就需要了。拿亮度滤镜来说,它需要配置它的实际亮度水平(-1全黑到1全白);对于降噪,噪音值为(0-1000);对于“去色”滤镜,要考虑阈值和距离值,等等。

现在你对于Fabric滤镜比较熟悉了,是时候跳出井底,创建自己的滤镜了!

创建滤镜的模版很直观,我们需要创建一个“类”,它定义了applyTo方法,可选的,我们也可以给滤镜定义toJSON方法(支持JSON序列化),定义initialize方法(支持可选参数)

fabric.Image.filters.Redify = fabric.util.createClass(fabric.Image.filters.BaseFilter, {

type: 'Redify',

/**

* Fragment source for the redify program

*/

fragmentSource: 'precision highp float;\n' +

'uniform sampler2D uTexture;\n' +

'varying vec2 vTexCoord;\n' +

'void main() {\n' +

'vec4 color = texture2D(uTexture, vTexCoord);\n' +

'color.g = 0;\n' +

'color.b = 0;\n' +

'gl_FragColor = color;\n' +

'}',

applyTo2d: function(options) {

var imageData = options.imageData,

data = imageData.data, i, len = data.length;

for (i = 0; i < len; i += 4) {

data[i + 1] = 0;

data[i + 2] = 0;

}

}

});

fabric.Image.filters.Redify.fromObject = fabric.Image.filters.BaseFilter.fromObject;

不需要深入研究这些代码,主要的操作发生在一个循环里,那里我们把每一个像素的绿色分量(data[i+1]) 和蓝色分量(data[i+2]) )设置为0,本质上是去除它们,标准的rgb三色中红色分量没有变,基本上把整个图像涂成了红色。如你所见,在过滤流水线上,applyTo方法被传入了一个包含图像数据的可选参数,从那里,我们可以迭代它的每一个像素(getImageData().data),按照我们需要的修改它们。如果浏览器支持WEBGL,滤镜可以在GPU上运行,为了实现这一点,你需要提供一个片元着色器,用来描述要处理的像素。在fabric定义了很多滤镜,你可以看看如何编写片元和顶点着色器的例子。

(译者注:在画布的图像数据中,每一个像素使用4个字节表示,分别是RGBA,代表红色、绿色、蓝色和alpha通道,其中alpha通道可以理解为透明度,上述代码中,把每个像素的绿色和蓝色分量都置为了0)

颜色

也行你更熟悉hex、RGB或RGBA颜色格式,但Fabric提供了一个坚实的颜色基础,帮助你更自然的表达自己。这里是在Fabric里定义颜色的几种方式:

new fabric.Color('#f55');

new fabric.Color('#123123');

new fabric.Color('356735');

new fabric.Color('rgb(100,0,100)');

new fabric.Color('rgba(10, 20, 30, 0.5)');

转换也很简单,toHex()转换颜色为16进制,toRgb()--转换为RGB,toRgba()--转换为带alpha通道的RGB:

new fabric.Color('#f55').toRgb(); // "rgb(255,85,85)"

new fabric.Color('rgb(100,100,100)').toHex(); // "646464"

new fabric.Color('fff').toHex(); // "FFFFFF"

对于颜色你不仅仅只能转换,也可以用一种颜色覆盖另外一种或者转换为灰度版本:

var redish = new fabric.Color('#f55');

var greenish = new fabric.Color('#5f5');

redish.overlayWith(greenish).toHex(); // "AAAA55"

redish.toGrayscale().toHex(); // "A1A1A1"

渐变

一种更具表现力的方法是通过渐变来处理颜色,渐变渐变使我们能够将一种颜色混合到另一种颜色中,创造一些惊人的图形效果。

Fabric通过定义在所有对象上的setGradient方法来支持渐变,就像设置一个对象的 "fill"值一样来调用setGradient('fill', { ... })方法,除非我们用渐变来填充对象而不是单一颜色:

var circle = new fabric.Circle({

left: 100,

top: 100,

radius: 50

});

circle.setGradient('fill', {

x1: 0,

y1: 0,

x2: 0,

y2: circle.height,

colorStops: {

0: '#000',

1: '#fff'

}

});

在上面的例子中,我们在位置100,100处创建了一个半径为50像素的圆,然后,我们将它的填充方式设置成一个横跨整个圆的高度,从黑色到白色的渐变。

传递给方法的参数是一个可选对象,它有两个坐标对(x1,y1和x2,y2),以及“colorStops”对象。坐标指示了渐变从哪里开始以及从哪里结束,colorStops指示了使用哪一种颜色渐变。你可以定义需要的多个颜色停止点,它们的范围都是从0到1(比如0,0.1,0.3,0.5,0.7,1),0表示渐变的开始,1代表结束。

坐标和对象的左上角有关系,因此原型的最高点是0,最低点是circle.height。setGradient 使用同样的方式计算宽度坐标(x1,x2)。

这是一个从左到右,从红到蓝的渐变例子:

circle.setGradient('fill', {

x1: 0,

y1: 0,

x2: circle.width,

y2: 0,

colorStops: {

0: "red",

1: "blue"

}

});

这是另一个5个停止点的彩虹渐变,颜色跨越20%的间隔:

circle.setGradient('fill', {

x1: 0,

y1: 0,

x2: circle.width,

y2: 0,

colorStops: {

0: "red",

0.2: "orange",

0.4: "yellow",

0.6: "green",

0.8: "blue",

1: "purple"

}

});

你能想出哪些更酷的版本呢?

未完待续

标签: #js透明度代码