mirror of
https://github.com/tsl0922/EPD-nRF5.git
synced 2025-12-06 15:42:48 +08:00
167 lines
5.9 KiB
JavaScript
167 lines
5.9 KiB
JavaScript
const bwrPalette = [
|
|
[0, 0, 0, 255],
|
|
[255, 255, 255, 255],
|
|
[255, 0, 0, 255]
|
|
]
|
|
|
|
function dithering(ctx, width, height, threshold, type) {
|
|
const bayerThresholdMap = [
|
|
[ 15, 135, 45, 165 ],
|
|
[ 195, 75, 225, 105 ],
|
|
[ 60, 180, 30, 150 ],
|
|
[ 240, 120, 210, 90 ]
|
|
];
|
|
|
|
const lumR = [];
|
|
const lumG = [];
|
|
const lumB = [];
|
|
for (let i=0; i<256; i++) {
|
|
lumR[i] = i*0.299;
|
|
lumG[i] = i*0.587;
|
|
lumB[i] = i*0.114;
|
|
}
|
|
const imageData = ctx.getImageData(0, 0, width, height);
|
|
const imageDataLength = imageData.data.length;
|
|
|
|
// Greyscale luminance (sets r pixels to luminance of rgb)
|
|
for (let i = 0; i <= imageDataLength; i += 4) {
|
|
imageData.data[i] = Math.floor(lumR[imageData.data[i]] + lumG[imageData.data[i+1]] + lumB[imageData.data[i+2]]);
|
|
}
|
|
|
|
const w = imageData.width;
|
|
let newPixel, err;
|
|
|
|
for (let currentPixel = 0; currentPixel <= imageDataLength; currentPixel+=4) {
|
|
if (type ==="none") { // No dithering
|
|
imageData.data[currentPixel] = imageData.data[currentPixel] < threshold ? 0 : 255;
|
|
} else if (type ==="bayer") { // 4x4 Bayer ordered dithering algorithm
|
|
var x = currentPixel/4 % w;
|
|
var y = Math.floor(currentPixel/4 / w);
|
|
var map = Math.floor( (imageData.data[currentPixel] + bayerThresholdMap[x%4][y%4]) / 2 );
|
|
imageData.data[currentPixel] = (map < threshold) ? 0 : 255;
|
|
} else if (type ==="floydsteinberg") { // Floyda€"Steinberg dithering algorithm
|
|
newPixel = imageData.data[currentPixel] < threshold ? 0 : 255;
|
|
err = Math.floor((imageData.data[currentPixel] - newPixel) / 16);
|
|
imageData.data[currentPixel] = newPixel;
|
|
|
|
imageData.data[currentPixel + 4 ] += err*7;
|
|
imageData.data[currentPixel + 4*w - 4 ] += err*3;
|
|
imageData.data[currentPixel + 4*w ] += err*5;
|
|
imageData.data[currentPixel + 4*w + 4 ] += err*1;
|
|
} else { // Bill Atkinson's dithering algorithm
|
|
newPixel = imageData.data[currentPixel] < threshold ? 0 : 255;
|
|
err = Math.floor((imageData.data[currentPixel] - newPixel) / 8);
|
|
imageData.data[currentPixel] = newPixel;
|
|
|
|
imageData.data[currentPixel + 4 ] += err;
|
|
imageData.data[currentPixel + 8 ] += err;
|
|
imageData.data[currentPixel + 4*w - 4 ] += err;
|
|
imageData.data[currentPixel + 4*w ] += err;
|
|
imageData.data[currentPixel + 4*w + 4 ] += err;
|
|
imageData.data[currentPixel + 8*w ] += err;
|
|
}
|
|
|
|
// Set g and b pixels equal to r
|
|
imageData.data[currentPixel + 1] = imageData.data[currentPixel + 2] = imageData.data[currentPixel];
|
|
}
|
|
|
|
ctx.putImageData(imageData, 0, 0);
|
|
}
|
|
|
|
// white: 1, black/red: 0
|
|
function canvas2bytes(canvas, step = 'bw', invert = false) {
|
|
const ctx = canvas.getContext("2d");
|
|
const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);
|
|
|
|
const arr = [];
|
|
let buffer = [];
|
|
|
|
for (let y = 0; y < canvas.height; y++) {
|
|
for (let x = 0; x < canvas.width; x++) {
|
|
const i = (canvas.width * y + x) * 4;
|
|
if (step === 'bw') {
|
|
buffer.push(imageData.data[i] === 0 && imageData.data[i+1] === 0 && imageData.data[i+2] === 0 ? 0 : 1);
|
|
} else {
|
|
buffer.push(imageData.data[i] > 0 && imageData.data[i+1] === 0 && imageData.data[i+2] === 0 ? 0 : 1);
|
|
}
|
|
|
|
if (buffer.length === 8) {
|
|
const data = parseInt(buffer.join(''), 2);
|
|
arr.push(invert ? ~data : data);
|
|
buffer = [];
|
|
}
|
|
}
|
|
}
|
|
return arr;
|
|
}
|
|
|
|
function getNearColorV2(color, palette) {
|
|
let minDistanceSquared = 255*255 + 255*255 + 255*255 + 1;
|
|
|
|
let bestIndex = 0;
|
|
for (let i = 0; i < palette.length; i++) {
|
|
let rdiff = (color[0] & 0xff) - (palette[i][0] & 0xff);
|
|
let gdiff = (color[1] & 0xff) - (palette[i][1] & 0xff);
|
|
let bdiff = (color[2] & 0xff) - (palette[i][2] & 0xff);
|
|
let distanceSquared = rdiff*rdiff + gdiff*gdiff + bdiff*bdiff;
|
|
if (distanceSquared < minDistanceSquared) {
|
|
minDistanceSquared = distanceSquared;
|
|
bestIndex = i;
|
|
}
|
|
}
|
|
return palette[bestIndex];
|
|
}
|
|
|
|
function updatePixel(imageData, index, color) {
|
|
imageData[index] = color[0];
|
|
imageData[index+1] = color[1];
|
|
imageData[index+2] = color[2];
|
|
imageData[index+3] = color[3];
|
|
}
|
|
|
|
function getColorErr(color1, color2, rate) {
|
|
const res = [];
|
|
for (let i = 0; i < 3; i++) {
|
|
res.push(Math.floor((color1[i] - color2[i]) / rate));
|
|
}
|
|
return res;
|
|
}
|
|
|
|
function updatePixelErr(imageData, index, err, rate) {
|
|
imageData[index] += err[0] * rate;
|
|
imageData[index+1] += err[1] * rate;
|
|
imageData[index+2] += err[2] * rate;
|
|
}
|
|
|
|
function ditheringCanvasByPalette(canvas, palette, type) {
|
|
palette = palette || bwrPalette;
|
|
|
|
const ctx = canvas.getContext('2d');
|
|
const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);
|
|
const w = imageData.width;
|
|
|
|
for (let currentPixel = 0; currentPixel <= imageData.data.length; currentPixel+=4) {
|
|
const newColor = getNearColorV2(imageData.data.slice(currentPixel, currentPixel+4), palette);
|
|
|
|
if (type === "bwr_floydsteinberg") {
|
|
const err = getColorErr(imageData.data.slice(currentPixel, currentPixel+4), newColor, 16);
|
|
|
|
updatePixel(imageData.data, currentPixel, newColor);
|
|
updatePixelErr(imageData.data, currentPixel +4, err, 7);
|
|
updatePixelErr(imageData.data, currentPixel + 4*w - 4, err, 3);
|
|
updatePixelErr(imageData.data, currentPixel + 4*w, err, 5);
|
|
updatePixelErr(imageData.data, currentPixel + 4*w + 4, err, 1);
|
|
} else {
|
|
const err = getColorErr(imageData.data.slice(currentPixel, currentPixel+4), newColor, 8);
|
|
|
|
updatePixel(imageData.data, currentPixel, newColor);
|
|
updatePixelErr(imageData.data, currentPixel +4, err, 1);
|
|
updatePixelErr(imageData.data, currentPixel +8, err, 1);
|
|
updatePixelErr(imageData.data, currentPixel +4 * w - 4, err, 1);
|
|
updatePixelErr(imageData.data, currentPixel +4 * w, err, 1);
|
|
updatePixelErr(imageData.data, currentPixel +4 * w + 4, err, 1);
|
|
updatePixelErr(imageData.data, currentPixel +8 * w, err, 1);
|
|
}
|
|
}
|
|
ctx.putImageData(imageData, 0, 0);
|
|
} |