用canvas 实现个图片三角化(LOW POLY)效果


Posted in Javascript onFebruary 18, 2016

之前无意中看到Ovilia 用threejs做了个LOW POLY,也就是图片平面三角化的效果,觉得很惊艳,然后就自己花了点时间尝试了一下。

我是没怎么用过threejs,所以就直接用canvas的2d绘图API来做,因为感觉似乎这效果也用不上threejs。

直接上demo先:http://whxaxes.github.io/canvas-test/src/Funny-demo/lowpoly/index.html (也可以在移动端看,不过因为计算量比较大,移动设备计算起来会比PC要多花些时间。)

做这种效果主要需要把图片三角化,以及对图片进行边缘化检测。这两个,第一个用到的delaunay三角化算法,第二个用到的sobel边缘检测算法。听起来偷挺高大上的,索性两个算法都有相应的开源组件可以直接拿来用:ironwallaby的delaunay组件 以及 Miguel Mota的sobel组件。

这两个算法sobel还好一点,delaunay就有点复杂了,待日后可以研究一下。不过目前只为做出个效果的话,还是可以用这些组件的。

两个最重要的组件都有了,剩下的事就很简单了:

先将图片绘制到canvas上:

canvas.width = (img.width > 800) ? 800 : img.width;
canvas.height = img.height * canvas.width/img.width;
ctx.drawImage(img, 0, 0, canvas.width, canvas.height);

然后获取到canvas的imgData,再通过sobel计算返回新的imgData

var imgData = ctx.getImageData(0, 0, canvas.width, canvas.height);
var newImgData = Sobel(imgData);

如果我们把newImgData放到canvas上,就会发现,彩色图片变成了这样的灰度图片:

用canvas 实现个图片三角化(LOW POLY)效果

由于上面说的那个Sobel组件不是很适合自己的用法,同时代码也有不恰当的地方,所以自己做了适当修改和优化,优化了循环方法,加快了运算速度,同时加入了回调函数。详见该项目github中的sobel.js文件

在Sobel方法中对imgData.data进行遍历的时候,会调用回调函数,在回调中把颜色值大于40(也就是灰度为rgb(40,40,40)以上的)的坐标点记录下来。然后随机获取一部分边缘点,再加入一些随机出来的坐标 以及 四个边角的坐标值。这样,我们就可以获取到我们需要的坐标点了

var imgData = ctx.getImageData(0, 0, canvas.width, canvas.height);
// 收集色值大于40的边缘像素点
var collectors = [];
Sobel(imgData , function(value , x , y){
if(value > 40){collectors.push([x , y]);}
});
// 添加一些随机点
for(var i=0;i<300;i++){particles.push([Math.random()*canvas.width , Math.random()*canvas.height]);}
// 添加随机边缘点,数量为边缘点数量除于50
var length = ~~(collectors.length/50), random;
for(var l=0;l<length;l++){
random = (Math.random()*collectors.length)<<0;
particles.push(collectors[random]);
collectors.splice(random , 1);
}
// 添加四顶点坐标
particles.push([0,0] , [0,canvas.height] , [canvas.width,0] , [canvas.width,canvas.height]);

获取到坐标点后,就可以通过delaunay组件计算,获取到拍好次序的三角坐标数组,对这些数组里的点进行连线,就可以出现这样的效果:

用canvas 实现个图片三角化(LOW POLY)效果

当然,我们要的效果不是连线,而是对所有三角形进行颜色填充,也就是获取三角形的三个坐标,然后计算出中心点的坐标,再根据中心点坐标在imgData里获取到相应的rgb的颜色值,然后填充到三角区域就可以了:

// 使用delaunay三角化获取三角坐标
var triangles = Delaunay.triangulate(particles);
var x1,x2,x3,y1,y2,y3,cx,cy;
for(var i=0;i < triangles.length; i+=3) {
x1 = particles[triangles[i]][0];
x2 = particles[triangles[i+1]][0];
x3 = particles[triangles[i+2]][0];
y1 = particles[triangles[i]][1];
y2 = particles[triangles[i+1]][1];
y3 = particles[triangles[i+2]][1];
// 获取三角形中心点坐标
cx = ~~((x1 + x2 + x3) / 3);
cy = ~~((y1 + y2 + y3) / 3);
// 获取中心点坐标的颜色值
index = (cy*imgData.width + cx)*4;
var color_r = imgData.data[index];
var color_g = imgData.data[index+1];
var color_b = imgData.data[index+2];
// 绘制三角形
ctx.save();
ctx.beginPath();
ctx.moveTo(x1, y1);
ctx.lineTo(x2, y2);
ctx.lineTo(x3, y3);
ctx.closePath();
ctx.fillStyle = "rgba("+color_r+","+color_g+","+color_b+",1)";
ctx.fill();
ctx.restore();
}

上面有一点要注意,获取到的中心点坐标一定要取整,才能够获取到正确的颜色参数,如果想着不取整,而是在获取rgb索引的时候再取整,获取到的颜色值就是错的。因为这样获取到的那个像素点就不是我们要的中心像素点。

颜色也获取到后,就是简单的连线,然后填充操作了,最后出来的效果就是:

用canvas 实现个图片三角化(LOW POLY)效果

用canvas 实现个图片三角化(LOW POLY)效果

以上内容给大家介绍了用canvas 实现个图片三角化(LOW POLY)效果 ,希望对大家有所帮助!

Javascript 相关文章推荐
根据对象的某一属性进行排序的js代码(如:name,age)
Aug 10 Javascript
JSON为什么那样红为什么要用json(另有洞天)
Dec 26 Javascript
jQuery监控文本框事件并作相应处理的方法
Apr 16 Javascript
JavaScript实现ASC转汉字及汉字转ASC的方法
Jan 23 Javascript
JS实现兼容各种浏览器的高级拖动方法完整实例【测试可用】
Jun 21 Javascript
JS树形菜单组件Bootstrap TreeView使用方法详解
Dec 21 Javascript
学习使用jQuery表单验证插件和日历插件
Feb 13 Javascript
深入理解ES6 Promise 扩展always方法
Sep 26 Javascript
webpack公共组件引用路径简化小技巧
Jun 15 Javascript
微信小程序如何访问公众号文章
Jul 08 Javascript
JQuery实现ul中添加LI和删除指定的Li元素功能完整示例
Oct 16 jQuery
JavaScript实现简单拖拽效果
Sep 15 Javascript
Js实现简单的小球运动特效
Feb 18 #Javascript
JavaScript与jQuery实现的闪烁输入效果
Feb 18 #Javascript
js实现简单的省市县三级联动效果实例
Feb 18 #Javascript
XML、HTML、CSS与JS的区别整理
Feb 18 #Javascript
jQuery插件实现适用于移动端的地址选择器
Feb 18 #Javascript
AngularJS 2.0新特性有哪些
Feb 18 #Javascript
JavaScript+canvas实现七色板效果实例
Feb 18 #Javascript
You might like
php过滤所有的空白字符(空格、全角空格、换行等)
2015/10/27 PHP
php官方微信接口大全(微信支付、微信红包、微信摇一摇、微信小店)
2015/12/21 PHP
PHP与Ajax相结合实现登录验证小Demo
2016/03/16 PHP
AJAX的使用方法详解
2017/04/29 PHP
在页面上点击任一链接时触发一个事件的代码
2007/04/07 Javascript
js 处理URL实用技巧
2010/11/23 Javascript
javascript学习笔记(十九) 节点的操作实现代码
2012/06/20 Javascript
javascript级联下拉列表实例代码(自写)
2013/05/10 Javascript
js网页版计算器的简单实现
2013/07/02 Javascript
Javascript中prototype属性实现给内置对象添加新的方法
2015/05/14 Javascript
使用Promise解决多层异步调用的简单学习心得
2016/05/17 Javascript
js实现简单的碰壁反弹效果
2016/08/30 Javascript
常用的js方法合集
2017/03/10 Javascript
javaScript实现复选框全选反选事件详解
2020/11/20 Javascript
JS运动特效之完美运动框架实例分析
2018/01/24 Javascript
vue的传参方式汇总和router使用技巧
2018/05/22 Javascript
Python常用的文件及文件路径、目录操作方法汇总介绍
2015/05/21 Python
matplotlib绘图实例演示标记路径
2018/01/23 Python
详解python3中zipfile模块用法
2018/06/18 Python
浅析python3中的os.path.dirname(__file__)的使用
2018/08/30 Python
详解Python字典小结
2018/10/20 Python
BP神经网络原理及Python实现代码
2018/12/18 Python
对python中的控制条件、循环和跳出详解
2019/06/24 Python
Python 面向对象静态方法、类方法、属性方法知识点小结
2020/03/09 Python
Django 用户登陆访问限制实例 @login_required
2020/05/13 Python
浅谈如何使用python抓取网页中的动态数据实现
2020/08/17 Python
PyTorch 中的傅里叶卷积实现示例
2020/12/11 Python
简单总结CSS3中视窗单位Viewport的常见用法
2016/02/04 HTML / CSS
纯css3实现宠物小鸡实例代码
2018/10/08 HTML / CSS
CSS3标注引用的出处和来源的方法
2020/02/25 HTML / CSS
Bandier官网:奢侈、时尚前卫的健身服装首选目的地
2020/07/05 全球购物
产品销售员岗位职责
2013/12/18 职场文书
考试退步检讨书
2014/01/15 职场文书
结婚保证书(卖身契)
2015/02/26 职场文书
大学新生入学感想
2015/08/07 职场文书
javascript的var与let,const之间的区别详解
2022/02/18 Javascript