前言:
此时我们对“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透明度代码