face-segmentation / index.js
Xenova's picture
Xenova HF staff
Create index.js
f12ecf3 verified
import { pipeline, env } from 'https://cdn.jsdelivr.net/npm/@xenova/[email protected]';
// Since we will download the model from the Hugging Face Hub, we can skip the local model check
env.allowLocalModels = false;
// Reference the elements that we will need
const status = document.getElementById('status');
const fileUpload = document.getElementById('upload');
const imageContainer = document.getElementById('container');
const example = document.getElementById('example');
const EXAMPLE_URL = 'https://i.imgur.com/mLvqQws.jpg';
// Create a new image segmentation pipeline
status.textContent = 'Loading model...';
const segmenter = await pipeline('image-segmentation', 'Xenova/face-parsing');
status.textContent = 'Ready';
example.addEventListener('click', (e) => {
e.preventDefault();
segment(EXAMPLE_URL);
});
fileUpload.addEventListener('change', function (e) {
const file = e.target.files[0];
if (!file) {
return;
}
const reader = new FileReader();
// Set up a callback when the file is loaded
reader.onload = e2 => segment(e2.target.result);
reader.readAsDataURL(file);
});
// Perform image segmentation
async function segment(img) {
imageContainer.innerHTML = '';
imageContainer.style.backgroundImage = `url(${img})`;
status.textContent = 'Analysing...';
const output = await segmenter(img);
status.textContent = '';
output.forEach(renderMask);
}
// Mapping of label to colour
const colours = [
[234, 76, 76], // red
[28, 180, 129], // sea green
[234, 155, 21], // orange
[67, 132, 243], // blue
[243, 117, 36], // orange-red
[145, 98, 243], // purple
[21, 178, 208], // cyan
[132, 197, 33], // lime
];
// Render a mask on the image
function renderMask({mask, label}, i) {
// Create new canvas
const canvas = document.createElement('canvas');
canvas.width = mask.width;
canvas.height = mask.height;
canvas.setAttribute('data-label', label);
// Create context and allocate buffer for pixel data
const context = canvas.getContext('2d');
const imageData = context.createImageData(canvas.width, canvas.height);
const pixelData = imageData.data;
// Choose colour based on index
const [r, g, b] = colours[i % colours.length];
// Fill mask with colour
for (let i = 0; i < pixelData.length; ++i) {
if (mask.data[i] !== 0) {
const offset = 4 * i;
pixelData[offset] = r; // red
pixelData[offset + 1] = g; // green
pixelData[offset + 2] = b; // blue
pixelData[offset + 3] = 255; // alpha (fully opaque)
}
}
// Draw image data to context
context.putImageData(imageData, 0, 0);
// Add canvas to container
imageContainer.appendChild(canvas);
}
// Clamp a value inside a range [min, max]
function clamp(x, min=0, max=1) {
return Math.max(Math.min(x, max), min)
}
// Attach hover event to image container
imageContainer.addEventListener('mousemove', e => {
const canvases = imageContainer.getElementsByTagName('canvas');
if (canvases.length === 0) return;
// Get bounding box
const bb = imageContainer.getBoundingClientRect();
// Get the mouse coordinates relative to the container
const mouseX = clamp((e.clientX - bb.left) / bb.width);
const mouseY = clamp((e.clientY - bb.top) / bb.height);
// Loop over all canvases
for (const canvas of canvases) {
const canvasX = canvas.width * mouseX;
const canvasY = canvas.height * mouseY;
// Get the pixel data of the mouse coordinates
const context = canvas.getContext('2d');
const pixelData = context.getImageData(canvasX, canvasY, 1, 1).data;
// Apply hover effect if not fully opaque
if (pixelData[3] < 255) {
canvas.style.opacity = 0.1;
} else {
canvas.style.opacity = 0.8;
status.textContent = canvas.getAttribute('data-label');
}
}
});
// Reset canvas opacities on mouse exit
imageContainer.addEventListener('mouseleave', e => {
const canvases = [...imageContainer.getElementsByTagName('canvas')];
if (canvases.length > 0) {
canvases.forEach(c => c.style.opacity = 0.6);
status.textContent = '';
}
})