利用canvas的getImageData,我们可以获取到一张图片每一个像素的信息,而通过对每一个像素信息的对比,我们就可以找到需要消去的像素点。比如下面这一张图片,如果我们想要扣去白色部分(粉色是body的背景颜色)。
let canvas = document.querySelector('#canvas'); let context = canvas.getContext('2d'); let img = document.createElement('img'); img.src = './head2.png'; img.onload = function () { canvas.height = img.height; canvas.width = img.width; context.drawImage(img, 0, 0); cutout(canvas, [255, 255, 255], 0.2); // 对白色进行抠除,容差为0.2 } function cutout(canvas, color, range = 0) { let context = canvas.getContext('2d'); let imageInfo = context.getImageData(0, 0, canvas.width, canvas.height); // pixiArr是一个数组,每四个数组元素代表一个像素点,这四个数组元素分别对应一个像素的r,g,b,a值。 let pixiArr = imageInfo.data; for (let i = 0; i < pixiArr.length; i += 4) { // 匹配到目标像素就将目标像素的alpha设为0 if (testColor([pixiArr[i], pixiArr[i + 1], pixiArr[i + 2]], color, range)) pixiArr[i + 3] = 0; } context.putImageData(imageInfo, 0, 0); } function testColor(current, target, range) { for (let i = 0; i < 3; i++) { if (!((1 - range) * target[i] <= current[i] && (1 + range) * target[i] >= current[i])) return false; } return true; }
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实现颜色容差抠图功能,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
稳了!魔兽国服回归的3条重磅消息!官宣时间再确认!
昨天有一位朋友在大神群里分享,自己亚服账号被封号之后居然弹出了国服的封号信息对话框。
这里面让他访问的是一个国服的战网网址,com.cn和后面的zh都非常明白地表明这就是国服战网。
而他在复制这个网址并且进行登录之后,确实是网易的网址,也就是我们熟悉的停服之后国服发布的暴雪游戏产品运营到期开放退款的说明。这是一件比较奇怪的事情,因为以前都没有出现这样的情况,现在突然提示跳转到国服战网的网址,是不是说明了简体中文客户端已经开始进行更新了呢?
更新日志
- 好薇2024《兵哥哥》1:124K黄金母盘[WAV+CUE]
- 胡歌.2006-珍惜(EP)【步升大风】【FLAC分轨】
- 洪荣宏.2014-拼乎自己看【华特】【WAV+CUE】
- 伊能静.1999-从脆弱到勇敢1987-1996精选2CD【华纳】【WAV+CUE】
- 刘亮鹭《汽车DJ玩主》[WAV+CUE][1.1G]
- 张杰《最接近天堂的地方》天娱传媒[WAV+CUE][1.1G]
- 群星《2022年度发烧天碟》无损黑胶碟 2CD[WAV+CUE][1.4G]
- 罗文1983-罗文甄妮-射雕英雄传(纯银AMCD)[WAV+CUE]
- 群星《亚洲故事香港纯弦》雨果UPMAGCD2024[低速原抓WAV+CUE]
- 群星《经典咏流传》限量1:1母盘直刻[低速原抓WAV+CUE]
- 庾澄庆1993《老实情歌》福茂唱片[WAV+CUE][1G]
- 许巍《在别处》美卡首版[WAV+CUE][1G]
- 林子祥《单手拍掌》华纳香港版[WAV+CUE][1G]
- 郑秀文.1997-我们的主题曲【华纳】【WAV+CUE】
- 群星.2001-生命因爱动听电影原创音乐AVCD【MEDIA】【WAV+CUE】