หากคุณต้องการให้ได้สีเฉลี่ยของพื้นที่สี่เหลี่ยมแทนที่จะเป็นสีของพิกเซลเดียวโปรดดูคำถามอื่นนี้:
👉 JavaScript - รับสีเฉลี่ยจากพื้นที่หนึ่งของภาพ
อย่างไรก็ตามทั้งสองจะทำในลักษณะที่คล้ายกันมาก:
🔍การรับสี / มูลค่าของพิกเซลเดียวจากรูปภาพหรือผ้าใบ
เพื่อให้ได้สีของพิกเซลเดียวก่อนอื่นคุณต้องวาดภาพนั้นลงบนผืนผ้าใบซึ่งคุณได้ทำไปแล้ว:
const image = document.getElementById('image');
const canvas = document.createElement('canvas');
const context = canvas.getContext('2d');
const width = image.width;
const height = image.height;
canvas.width = width;
canvas.height = height;
context.drawImage(image, 0, 0, width, height);
จากนั้นรับค่าของพิกเซลเดียวดังนี้:
const data = context.getImageData(X, Y, 1, 1).data;
🚀เร่งความเร็วโดยรับ ImageData ทั้งหมดในครั้งเดียว
คุณต้องใช้CanvasRenderingContext2D.getImageData ()เดียวกันนี้เพื่อรับค่าของภาพทั้งหมดซึ่งคุณทำได้โดยการเปลี่ยนพารามิเตอร์ที่สามและสี่ ลายเซ็นของฟังก์ชันนั้นคือ:
ImageData ctx.getImageData(sx, sy, sw, sh);
sx
: พิกัด x ของมุมบนซ้ายของสี่เหลี่ยมผืนผ้าที่จะดึง ImageData
sy
: พิกัด y ของมุมบนซ้ายของสี่เหลี่ยมผืนผ้าที่จะดึง ImageData
sw
: ความกว้างของรูปสี่เหลี่ยมผืนผ้าที่จะดึง ImageData
sh
: ความสูงของรูปสี่เหลี่ยมผืนผ้าที่จะดึง ImageData
คุณสามารถเห็นมันส่งกลับImageData
วัตถุสิ่งที่เป็น ส่วนสำคัญที่นี่คือวัตถุนั้นมี.data
คุณสมบัติที่มีค่าพิกเซลทั้งหมดของเรา
อย่างไรก็ตามโปรดทราบว่า.data
คุณสมบัติคือ 1 มิติUint8ClampedArray
ซึ่งหมายความว่าส่วนประกอบทั้งหมดของพิกเซลถูกแบนดังนั้นคุณจะได้รับสิ่งที่มีลักษณะดังนี้:
สมมติว่าคุณมีภาพ 2x2 ดังนี้:
RED PIXEL | GREEN PIXEL
BLUE PIXEL | TRANSPARENT PIXEL
จากนั้นคุณจะได้รับสิ่งนี้:
[ 255, 0, 0, 255, 0, 255, 0, 255, 0, 0, 255, 255, 0, 0, 0, 0 ]
| RED PIXEL | GREEN PIXEL | BLUE PIXEL | TRANSPAERENT PIXEL |
| 1ST PIXEL | 2ND PIXEL | 3RD PIXEL | 4TH PIXEL |
เนื่องจากการโทรgetImageData
เป็นการดำเนินการที่ช้าคุณสามารถเรียกใช้เพียงครั้งเดียวเพื่อรับข้อมูลของภาพทั้งหมด ( sw
= ความกว้างของภาพ, sh
= ความสูงของภาพ)
จากนั้นในตัวอย่างข้างต้นถ้าคุณต้องการที่จะเข้าถึงส่วนประกอบของTRANSPARENT PIXEL
, ที่อยู่, หนึ่งที่ตำแหน่งx = 1, y = 1
ของภาพจินตนาการนี้คุณจะพบว่าดัชนีครั้งแรกi
ในของมันImageData
's data
คุณสมบัติดังนี้
const i = (y * imageData.width + x) * 4;
✨มาดูกันเลย
const solidColor = document.getElementById('solidColor');
const alphaColor = document.getElementById('alphaColor');
const solidWeighted = document.getElementById('solidWeighted');
const solidColorCode = document.getElementById('solidColorCode');
const alphaColorCode = document.getElementById('alphaColorCode');
const solidWeightedCOde = document.getElementById('solidWeightedCode');
const brush = document.getElementById('brush');
const image = document.getElementById('image');
const canvas = document.createElement('canvas');
const context = canvas.getContext('2d');
const width = image.width;
const height = image.height;
const BRUSH_SIZE = brush.offsetWidth;
const BRUSH_CENTER = BRUSH_SIZE / 2;
const MIN_X = image.offsetLeft + 4;
const MAX_X = MIN_X + width - 1;
const MIN_Y = image.offsetTop + 4;
const MAX_Y = MIN_Y + height - 1;
canvas.width = width;
canvas.height = height;
context.drawImage(image, 0, 0, width, height);
const imageDataData = context.getImageData(0, 0, width, height).data;
function sampleColor(clientX, clientY) {
if (clientX < MIN_X || clientX > MAX_X || clientY < MIN_Y || clientY > MAX_Y) {
requestAnimationFrame(() => {
brush.style.transform = `translate(${ clientX }px, ${ clientY }px)`;
solidColorCode.innerText = solidColor.style.background = 'rgb(0, 0, 0)';
alphaColorCode.innerText = alphaColor.style.background = 'rgba(0, 0, 0, 0.00)';
solidWeightedCode.innerText = solidWeighted.style.background = 'rgb(0, 0, 0)';
});
return;
}
const imageX = clientX - MIN_X;
const imageY = clientY - MIN_Y;
const i = (imageY * width + imageX) * 4;
const R = imageDataData[i];
const G = imageDataData[i + 1];
const B = imageDataData[i + 2];
const A = imageDataData[i + 3] / 255;
const iA = 1 - A;
const wR = (R * A + 255 * iA) | 0;
const wG = (G * A + 255 * iA) | 0;
const wB = (B * A + 255 * iA) | 0;
requestAnimationFrame(() => {
brush.style.transform = `translate(${ clientX }px, ${ clientY }px)`;
solidColorCode.innerText = solidColor.style.background
= `rgb(${ R }, ${ G }, ${ B })`;
alphaColorCode.innerText = alphaColor.style.background
= `rgba(${ R }, ${ G }, ${ B }, ${ A.toFixed(2) })`;
solidWeightedCode.innerText = solidWeighted.style.background
= `rgb(${ wR }, ${ wG }, ${ wB })`;
});
}
document.onmousemove = (e) => sampleColor(e.clientX, e.clientY);
sampleColor(MIN_X, MIN_Y);
body {
margin: 0;
height: 100vh;
display: flex;
flex-direction: row;
align-items: center;
justify-content: center;
cursor: none;
font-family: monospace;
overflow: hidden;
}
#image {
border: 4px solid white;
border-radius: 2px;
box-shadow: 0 0 32px 0 rgba(0, 0, 0, .25);
width: 150px;
box-sizing: border-box;
}
#brush {
position: absolute;
top: 0;
left: 0;
pointer-events: none;
width: 1px;
height: 1px;
mix-blend-mode: exclusion;
border-radius: 100%;
}
#brush::before,
#brush::after {
content: '';
position: absolute;
background: magenta;
}
#brush::before {
top: -16px;
left: 0;
height: 33px;
width: 100%;
}
#brush::after {
left: -16px;
top: 0;
width: 33px;
height: 100%;
}
#samples {
position: relative;
list-style: none;
padding: 0;
width: 250px;
}
#samples::before {
content: '';
position: absolute;
top: 0;
left: 27px;
width: 2px;
height: 100%;
background: black;
border-radius: 1px;
}
#samples > li {
position: relative;
display: flex;
flex-direction: column;
justify-content: center;
padding-left: 56px;
}
#samples > li + li {
margin-top: 8px;
}
.sample {
position: absolute;
top: 50%;
left: 16px;
transform: translate(0, -50%);
display: block;
width: 24px;
height: 24px;
border-radius: 100%;
box-shadow: 0 0 16px 4px rgba(0, 0, 0, .25);
margin-right: 8px;
}
.sampleLabel {
font-weight: bold;
margin-bottom: 8px;
}
.sampleCode {
}
<img id="image" src="" >
<div id="brush"></div>
<ul id="samples">
<li>
<span class="sample" id="solidColor"></span>
<div class="sampleLabel">solidColor</div>
<div class="sampleCode" id="solidColorCode">rgb(0, 0, 0)</div>
</li>
<li>
<span class="sample" id="alphaColor"></span>
<div class="sampleLabel">alphaColor</div>
<div class="sampleCode" id="alphaColorCode">rgba(0, 0, 0, 0.00)</div>
</li>
<li>
<span class="sample" id="solidWeighted"></span>
<div class="sampleLabel">solidWeighted (with white)</div>
<div class="sampleCode" id="solidWeightedCode">rgb(0, 0, 0)</div>
</li>
</ul>
⚠️หมายเหตุฉันกำลังใช้ URI ข้อมูลขนาดเล็กเพื่อหลีกเลี่ยงCross-Origin
ปัญหาหากฉันใส่รูปภาพภายนอกหรือคำตอบที่ใหญ่กว่าที่อนุญาตหากฉันพยายามใช้ URI ข้อมูลที่ยาวขึ้น
🕵️สีเหล่านี้ดูแปลก ๆ ใช่ไหม?
หากคุณเลื่อนเคอร์เซอร์ไปรอบ ๆ ขอบของรูปดอกจันคุณจะเห็นบางครั้งavgSolidColor
เป็นสีแดง แต่พิกเซลที่คุณสุ่มตัวอย่างจะเป็นสีขาว นั่นเป็นเพราะแม้ว่าR
ส่วนประกอบของพิกเซลนั้นอาจจะสูง แต่ช่องอัลฟาก็ต่ำดังนั้นสีจึงเป็นสีแดงที่เกือบโปร่งใส แต่avgSolidColor
ไม่สนใจสิ่งนั้น
ในทางกลับกันavgAlphaColor
ดูเป็นสีชมพู นั่นไม่ใช่ความจริงมันดูเป็นสีชมพูเพราะตอนนี้เราใช้ช่องอัลฟาซึ่งทำให้เป็นแบบกึ่งโปร่งใสและทำให้เราเห็นพื้นหลังของหน้าซึ่งในกรณีนี้จะเป็นสีขาว
🎨สีอัลฟ่าถ่วงน้ำหนัก
แล้วเราจะแก้ไขปัญหานี้ได้อย่างไร? ปรากฎว่าเราต้องใช้ช่องอัลฟาและผกผันเป็นน้ำหนักในการคำนวณส่วนประกอบของตัวอย่างใหม่ของเราในกรณีนี้จะรวมเข้ากับสีขาวเนื่องจากเป็นสีที่เราใช้เป็นพื้นหลัง
นั่นหมายความว่าถ้าพิกเซลอยู่R, G, B, A
ที่ไหนA
ในช่วงเวลา[0, 1]
เราจะคำนวณค่าผกผันของช่องอัลฟาiA
และส่วนประกอบของตัวอย่างที่ถ่วงน้ำหนักเป็น:
const iA = 1 - A;
const wR = (R * A + 255 * iA) | 0;
const wG = (G * A + 255 * iA) | 0;
const wB = (B * A + 255 * iA) | 0;
สังเกตว่าพิกเซลมีความโปร่งใสมากเพียงใด ( A
ใกล้ 0) สีก็จะยิ่งอ่อนลง