html5canvas曲线图例子(html5利用canvas实现颜色容差抠图功能)
html5canvas曲线图例子
html5利用canvas实现颜色容差抠图功能利用canvas的getImageData,我们可以获取到一张图片每一个像素的信息,而通过对每一个像素信息的对比,我们就可以找到需要消去的像素点。比如下面这一张图片,如果我们想要扣去白色部分(粉色是body的背景颜色)。
let canvas = document.querySelector('#canvas'); let context = canvas.getContext('2d'); let img = document.createElement('img'); img.src = './head2.jpg" alt="html5canvas曲线图例子(html5利用canvas实现颜色容差抠图功能)" border="0" />
testColor(current, target, range) 方法三个参数分别为 待检测像素点的rgb数组 、 目标像素点的rgb数组 和 容差范围 ,这里的容差只是简单用r、g、b的值分别乘以(1 + range)和(1 - range)来计算并对比,不同的容差参数会得到不同的效果↓
range = 0.095
range = 0.1
range = 0.2
当然对于底色是标准的纯色的图片就不需要容差了。
边界处理
但是有时候我们希望有一个边界,让抠图操作不对边界内部的像素造成影响。比如上面的图片,我们希望不会对人物头像内部的像素造成影响。 如果我们一行一行来看,是不是只要在碰到不是边界像素的时候停止操作,就可以达到效果了呢?
我们对每一行分别进行扫描,定义一个左指针 left 指向这一行的第一个像素,定义一个右指针 right 指向这一行的最后一个像素,并用一个 leftF 标识左边是否碰到边界,一个 rightF 标识右边是否碰到边界,当没碰到边界时指针就一直向内收缩,直到两个指针都碰到边界或者左右指针重合就跳到下一行,直到所有行都扫描完毕。
function cutout(canvas, color, range = 0) { let context = canvas.getContext('2d'); let imageInfo = context.getImageData(0, 0, canvas.width, canvas.height); let pixiArr = imageInfo.data; for (let row = 0; row < canvas.height; row++) { let left = row * 4 * canvas.width; // 指向行首像素 let right = left + 4 * canvas.width - 1 - 3; // 指向行尾像素 let leftF = false; // 左指针是否碰到边界的标识 let rightF = false; // 右指针是否碰到边界的标识 while (!leftF || !rightF) { // 当左右指针都为true,即都碰到边界时结束 if (!leftF) { if (testColor([pixiArr[left], pixiArr[left + 1], pixiArr[left + 2]], color, range)) { pixiArr[left + 3] = 0; // 此像素的alpha设为0 left += 4; // 移到下一个像素 } else leftF = true; // 碰到边界 } if (!rightF) { if (testColor([pixiArr[right], pixiArr[right + 1], pixiArr[right + 2]], color, range)) { pixiArr[right + 3] = 0; right -= 4; } else rightF = true; } if (left == right) { // 左右指针重合 leftF = true; rightF = true; }; } } context.putImageData(imageInfo, 0, 0); }
虽然大概完成了我们的需求,但是看一下上面头发那为啥会多了一块白色
因为我们仅仅只进行了行扫描,当左指针碰到头发时就会停止扫描,但是头发弧度里面的就无法被扫描到了,我们还需要进行列扫描,改造一下上面的方法:
function cutout(canvas, color, range = 0) { let context = canvas.getContext('2d'); let imageInfo = context.getImageData(0, 0, canvas.width, canvas.height); let pixiArr = imageInfo.data; for (let row = 0; row < canvas.height; row++) { let left = row * 4 * canvas.width; let right = left + 4 * canvas.width - 1 - 3; let leftF = false; let rightF = false; while (!leftF || !rightF) { if (!leftF) { if (testColor([pixiArr[left], pixiArr[left + 1], pixiArr[left + 2]], color, range)) { pixiArr[left + 3] = 0; left += 4; } else leftF = true; } if (!rightF) { if (testColor([pixiArr[right], pixiArr[right + 1], pixiArr[right + 2]], color, range)) { pixiArr[right + 3] = 0; right -= 4; } else rightF = true; } if (left == right) { leftF = true; rightF = true; }; } } // 同理进行列扫描 for (let col = 0; col < canvas.width; col++) { let top = col * 4; // 指向列头 let bottom = top + (canvas.height - 2) * canvas.width * 4 + canvas.width * 4; // 指向列尾 let topF = false; let bottomF = false; while (!topF || !bottomF) { if (!topF) { if (testColor([pixiArr[top], pixiArr[top + 1], pixiArr[top + 2]], color, range)) { pixiArr[top + 3] = 0; top += canvas.width * 4; } else topF = true; } if (!bottomF) { if (testColor([pixiArr[bottom], pixiArr[bottom + 1], pixiArr[bottom + 2]], color, range)) { pixiArr[bottom + 3] = 0; bottom -= canvas.width * 4; } else bottomF = true; } if (top == bottom) { topF = true; bottomF = true; }; } } context.putImageData(imageInfo, 0, 0); }
至于top和bottom为啥是那样计算画个矩阵图大概就知道了。
处理后↓
其实还可以先将 pixiArr 包装为以一个像素点为单位的矩阵
[ [{r: 255, g: 255, b: 255, a: 1}, {r: 255, g: 255, b: 255, a: 1}, {r: 255, g: 255, b: 255, a: 1}], [{r: 255, g: 255, b: 255, a: 1}, {r: 255, g: 255, b: 255, a: 1}, {r: 255, g: 255, b: 255, a: 1}] [{r: 255, g: 255, b: 255, a: 1}, {r: 255, g: 255, b: 255, a: 1}, {r: 255, g: 255, b: 255, a: 1}] ]
处理后计算像素下标也就会更简单,列扫描时直接先将这个矩阵旋转,再用行扫描处理一遍就行了。这样处理pixiArr也有利于进一步对算法进行优化。
上述方法虽然大概完成了抠图效果,但是这种简单处理还会有许多情况没有考虑到。
比如右边头发这里是行扫描和列扫描都无法触碰到的区域↓
下面的衣服也因为颜色和底色一样且没有边界在列扫描中被直接抹去了↓
最后
对于主体和底色区分度很大的图片来说,最开始的那种方法就已经够用了。这篇中我采用的容差和边界处理算法的优化空间还很大,但是它们是非常容易实现与理解的,这篇权当做一个引子,各位完全可以根据自己的能力继续去实现边界与容差算法。
总结
以上所述是小编给大家介绍的html5利用canvas实现颜色容差抠图功能,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对开心学习网网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!
- canvas指定区域生成图片(canvas实现图片镜像翻转的2种方式)
- html判断canvas已经绘制过了(html2canvas截图空白问题的解决)
- canvas绘制渐变图形(Canvas实现放大镜效果完整案例分析附代码)
- canvas图片填充位置(手摸手教你用canvas实现给图片添加平铺水印的实现)
- html 设置canvas的背景图可缩放(浅析canvas元素的html尺寸和css尺寸对元素视觉的影响)
- canvas画布多次渲染失败(深入了解canvas在移动端绘制模糊的问题解决)
- canvas 绘图解决方案(高清屏下canvas重置尺寸引发的问题的解决)
- canvas俄罗斯方块(JavaScript canvas实现俄罗斯方块游戏)
- 使用canvas的好坏(关于canvas绘制模糊问题的解决方法)
- canvas处理图片风格(html2 canvas生成清晰的图片实现打印功能)
- 使用canvas画个正方形(canvas绘制树形结构可视图形的实现)
- canvas心形水波(Canvas波浪花环的示例代码)
- html5canvas画图有什么用(Html5基于canvas实现电子签名并生成PDF文档)
- 优秀的canvas背景特效网站(基于canvas实现超炫酷的流水灯效果)
- canvas如何开启(canvas实现手机的手势解锁的步骤详细)
- canvas怎么获取内容(详解canvas.toDataURL报错的解决方案全都在这了)
- 书法欣赏 宋.志南诗《绝句》(宋.志南诗绝句)
- 每周一首古诗 《绝句》(每周一首古诗绝句)
- 蓝色代表什么(蓝色代表什么性格的人)
- 红色代表什么(红色代表什么情感和含义)
- 南宋志南和尚绝句 杨柳风似庙中来(南宋志南和尚绝句)
- 今天要穿什么颜色(今天要穿什么颜色的衣服最吉利)
热门推荐
- C#递归的应用
- centos中安装sql图(CentOS 7.3上SQL Server vNext CTP 1.2安装教程)
- python中list怎么用(详解python中list的使用)
- js 出现cannot find function(Fatal error: Call to a member function read on a non-object in 错误解决方法)
- python3json序列化(Python3.5 Json与pickle实现数据序列化与反序列化操作示例)
- jenkins 构建docker镜像(docker搭建jenkins+maven代码构建部署平台)
- python发送微信消息脚本(python实现给微信指定好友定时发送消息)
- python发送钉钉消息(钉钉群自定义机器人消息Python封装的实例)
- 如何解决AJAX 的缓存
- centosftp服务器的配置(CentOS6.9中搭建FTP服务器的方法)
排行榜
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9