需求:
需要改变图片中某一物体的颜色,该物体是纯色;
鼠标点击哪个物体,哪个物体的颜色变为指定的颜色,利用canvas实现。
演示案例
代码Demo
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<title>Color Replacement</title>
<style>
canvas {
border: 1px solid black;
}
</style>
</head>
<body>
<input type="color" id="newColor" name="newColor" />
<canvas id="myCanvas" width="375" height="397"></canvas>
<script>
const canvas = document.getElementById('myCanvas');
const ctx = canvas.getContext('2d', { willReadFrequently: true });
const img = new Image();
img.src =
'';
img.onload = function () {
ctx.drawImage(img, 0, 0, canvas.width, canvas.height);
};
// 替换颜色
function replaceColor(clickX, clickY, newColor) {
const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);
const data = imageData.data;
// debugger;
// 点击的颜色值
const index = parseInt((clickY * canvas.width + clickX) * 4);
const oldColor = [data[index], data[index + 1], data[index + 2]];
// 将颜色转成rgb
const result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(newColor.slice(1));
const newRGB = result ? [parseInt(result[1], 16), parseInt(result[2], 16), parseInt(result[3], 16)] : null;
if (!newRGB) return;
floodFill(imageData, clickX, clickY, oldColor, newRGB);
ctx.putImageData(imageData, 0, 0);
}
// 填充颜色
function floodFill(imageData, x, y, oldColor, newColor) {
let stack = [[x, y]];
const data = imageData.data;
let includesPoints = [];
while (stack.length) {
let point = stack.pop();
let index = parseInt((point[1] * canvas.width + point[0]) * 4);
if (data[index] === oldColor[0] && data[index + 1] === oldColor[1] && data[index + 2] === oldColor[2]) {
data[index] = newColor[0];
data[index + 1] = newColor[1];
data[index + 2] = newColor[2];
data[index + 3] = 255;
[
[0, -1],
[1, 0],
[0, 1],
[-1, 0],
].forEach(([dx, dy]) => {
let newX = point[0] + dx;
let newY = point[1] + dy;
const pointToString = newX + ',' + newY;
if (newX >= 0 && newX < canvas.width && newY >= 0 && newY < canvas.height) {
stack.push([newX, newY]);
}
});
}
}
}
canvas.addEventListener('click', function (event) {
const rect = canvas.getBoundingClientRect();
const clickX = event.clientX - rect.left;
const clickY = event.clientY - rect.top;
// debugger;
let newColor = document.getElementById('newColor').value;
replaceColor(Math.round(clickX), Math.round(clickY), newColor);
});
</script>
</body>
</html>
后续
确定点击点四周相同颜色值的算法有待优化