Spaces:
Running
Running
File size: 6,587 Bytes
1f2f6bf 00fddb0 1f2f6bf 00fddb0 1f2f6bf 00fddb0 1f2f6bf 00fddb0 1f2f6bf | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 | import { useEffect, useMemo, useState } from "react";
const MAX_SIZE: number = 500;
function getBoundedDimensions(width: number, height: number): [number, number] {
if (width <= MAX_SIZE && height <= MAX_SIZE) {
return [width, height];
}
const scale = Math.min(MAX_SIZE / width, MAX_SIZE / height);
const boundedWidth = Math.max(1, Math.round(width * scale));
const boundedHeight = Math.max(1, Math.round(height * scale));
return [boundedWidth, boundedHeight];
}
async function getImageData(imageUrl: string): Promise<ImageData> {
const image = await new Promise<HTMLImageElement>((resolve, reject) => {
const img = new Image();
img.crossOrigin = "anonymous";
img.onload = () => resolve(img);
img.onerror = reject;
img.src = imageUrl;
})
const sourceWidth = image.naturalWidth || image.width;
const sourceHeight = image.naturalHeight || image.height;
const [targetWidth, targetHeight] = getBoundedDimensions(sourceWidth, sourceHeight);
const canvas = document.createElement("canvas")
canvas.width = targetWidth;
canvas.height = targetHeight;
const ctx = canvas.getContext("2d");
if (!ctx) {
throw new Error("Failed to get canvas context");
}
ctx.drawImage(image, 0, 0, targetWidth, targetHeight);
return ctx.getImageData(0, 0, canvas.width, canvas.height);
}
function getImageUrl(imageData: ImageData): string {
const canvas = document.createElement("canvas");
canvas.width = imageData.width;
canvas.height = imageData.height;
const ctx = canvas.getContext("2d")
if (!ctx) {
throw new Error("Failed to get canvas context");
}
ctx.putImageData(imageData, 0, 0);
return canvas.toDataURL("image/png");
}
function convertToGrayscale(imageData: ImageData): ImageData {
const output = new ImageData(
new Uint8ClampedArray(imageData.data),
imageData.width,
imageData.height,
);
const data = output.data;
for (let i = 0; i < data.length; i += 4) {
const r = data[i];
const g = data[i + 1];
const b = data[i + 2];
const gray = 0.299 * r + 0.587 * g + 0.114 * b;
data[i] = data[i + 1] = data[i + 2] = gray;
}
return output;
}
function convolve(imageData: ImageData, kernel: number[][] | number[][][]): ImageData {
if (Array.isArray(kernel[0][0])) {
// 3D kernel (color)
return convolveColor(imageData, kernel as number[][][]);
} else {
// 2D kernel (grayscale)
return convolveGray(imageData, kernel as number[][]);
}
}
function convolveGray(image: ImageData, kernel: number[][]): ImageData {
const kernelWidth = kernel[0].length;
const kernelHeight = kernel.length;
const width = image.width;
const height = image.height;
const inputData = image.data;
const outputWidth = width - kernelWidth + 1;
const outputHeight = height - kernelHeight + 1;
const outputData = new Uint8ClampedArray(outputWidth * outputHeight * 4);
for (let y = 0; y < outputHeight; ++y) {
for (let x = 0; x < outputWidth; ++x) {
// dot product
let sum = 0;
for (let ky = 0; ky < kernelHeight; ++ky) {
for (let kx = 0; kx < kernelWidth; ++kx) {
const pixelIndex = ((y + ky) * width + (x + kx)) * 4;
const pixelValue = inputData[pixelIndex];
const kernelValue = kernel[ky][kx];
sum += pixelValue * kernelValue;
}
}
const outputIndex = (y * outputWidth + x) * 4;
const clampedValue = Math.min(Math.max(sum, 0), 255);
outputData[outputIndex] = clampedValue; // R
outputData[outputIndex + 1] = clampedValue; // G
outputData[outputIndex + 2] = clampedValue; // B
outputData[outputIndex + 3] = 255; // A
}
}
return new ImageData(outputData, outputWidth, outputHeight);
}
function convolveColor(image: ImageData, kernel: number[][][]): ImageData {
const kernelWidth = kernel[0][0].length;
const kernelHeight = kernel[0].length;
const width = image.width;
const height = image.height;
const inputData = image.data;
const outputWidth = width - kernelWidth + 1;
const outputHeight = height - kernelHeight + 1;
const outputData = new Uint8ClampedArray(outputWidth * outputHeight * 4);
for (let y = 0; y < outputHeight; ++y) {
for (let x = 0; x < outputWidth; ++x) {
// dot product over 3 channels
let sum = 0;
for (let ky = 0; ky < kernelHeight; ++ky) {
for (let kx = 0; kx < kernelWidth; ++kx) {
const pixelIndex = ((y + ky) * width + (x + kx)) * 4;
const r = inputData[pixelIndex];
const g = inputData[pixelIndex + 1];
const b = inputData[pixelIndex + 2];
const kernelR = kernel[0][ky][kx];
const kernelG = kernel[1][ky][kx];
const kernelB = kernel[2][ky][kx];
sum += r * kernelR + g * kernelG + b * kernelB;
}
}
const outputIndex = (y * outputWidth + x) * 4;
const clampedValue = Math.min(Math.max(sum, 0), 255);
outputData[outputIndex] = clampedValue; // R
outputData[outputIndex + 1] = clampedValue; // G
outputData[outputIndex + 2] = clampedValue; // B
outputData[outputIndex + 3] = 255; // A
}
}
return new ImageData(outputData, outputWidth, outputHeight);
}
export default function useConvolutionProcessing(
rawInputImage: string,
kernel: number[][][] | number[][],
): [string | null, string | null] {
const useColor = Array.isArray(kernel[0][0]); // true if 3D kernel, false if 2D kernel
const [rawImageData, setRawImageData] = useState<ImageData | null>(null);
// extract input image data (array)
useEffect(() => {
let cancelled = false;
async function processImage() {
const imageData = await getImageData(rawInputImage);
if (!cancelled) {
setRawImageData(imageData);
}
}
processImage();
return () => {
cancelled = true;
}
}, [rawInputImage]);
const processedImageData = useMemo(() => {
if (!rawImageData) return null;
return useColor ? rawImageData : convertToGrayscale(rawImageData);
}, [rawImageData, useColor]);
const outputImageData = useMemo(() => {
if (!processedImageData) return null;
return convolve(processedImageData, kernel);
}, [processedImageData, kernel]);
const inputImage = useMemo(() => {
if (!processedImageData) return null;
return getImageUrl(processedImageData);
}, [processedImageData]);
const outputImage = useMemo(() => {
if (!outputImageData) return null;
return getImageUrl(outputImageData);
}, [outputImageData]);
return [inputImage, outputImage];
}
|