Автоматически обрезать холст HTML5 до содержимого
Скажем, это мой холст, со злобным лицом, нарисованным на нем. Я хочу использовать toDataURL()
для экспорта моего злого лица в виде PNG; однако весь холст растеризуется, включая "пробел" между злым лицом и краями холста.
+---------------+
| |
| |
| (.Y. ) |
| /_ |
| \____/ |
| |
| |
+---------------+
Каков наилучший способ обрезать/обрезать/обрезать мой холст до его содержимого, поэтому мой PNG не больше, чем граничная рамка лица, как показано ниже? Лучший способ - это масштабирование холста, но предположим, что содержимое динамическое...? Я уверен, что для этого должно быть простое решение, но оно ускользает от меня, с большим количеством Googling.
+------+
|(.Y. )|
| /_ |
|\____/|
+------+
Спасибо!
Ответы
Ответ 1
function cropImageFromCanvas(ctx, canvas) {
var w = canvas.width,
h = canvas.height,
pix = {x:[], y:[]},
imageData = ctx.getImageData(0,0,canvas.width,canvas.height),
x, y, index;
for (y = 0; y < h; y++) {
for (x = 0; x < w; x++) {
index = (y * w + x) * 4;
if (imageData.data[index+3] > 0) {
pix.x.push(x);
pix.y.push(y);
}
}
}
pix.x.sort(function(a,b){return a-b});
pix.y.sort(function(a,b){return a-b});
var n = pix.x.length-1;
w = pix.x[n] - pix.x[0];
h = pix.y[n] - pix.y[0];
var cut = ctx.getImageData(pix.x[0], pix.y[0], w, h);
canvas.width = w;
canvas.height = h;
ctx.putImageData(cut, 0, 0);
var image = canvas.toDataURL();
var win=window.open(image, '_blank');
win.focus();
}
Ответ 2
Если я правильно понял, вы хотите "обрезать" все ваши изображения/чертежи и настроить холст на этот размер (например, если вы делаете команду "trim" в Photoshop).
Вот как я это сделаю.
-
Запустите все пиксели холста, проверяя, есть ли их альфа-компонент > 0 (это означает, что в этом пикселе что-то нарисовано). Альтернативно вы можете проверить значения r, g, b, если ваш фон холста заполнен сплошным цветом, например.
-
Получить те координаты верхнего левого пикселя непустые, а также для самого нижнего правого. Таким образом, вы получите координаты воображаемого "прямоугольника", содержащего область холста, которая не является пустой.
-
Сохраните эту область пикселей.
-
Измените размер вашего холста на его новые размеры (те области, которые мы получили на шаге 2.)
-
Вставьте сохраненную область обратно в холст.
Et, voilá:)
Приблизительная пиксельная точка довольно медленная в зависимости от размера вашего холста (если ее огромная она может занять некоторое время). Есть несколько оптимизаций для работы с необработанной пиксельной таблицей (я думаю, что есть статья об этой теме в MDN), я предлагаю вам об этом Google.
Я подготовил небольшой эскиз в jsFiddle, который вы можете использовать в качестве отправной точки для своего кода.
Рабочий пример в jsFiddle
Надеюсь, я помог тебе.
C:.
Ответ 3
Здесь проголосовали лучшие, а также реализации, которые я нашел в Интернете, обрезать один дополнительный пиксель, что было очень заметно при попытке обрезать текст из канвы. Я написал свой, который работал лучше для меня:
var img = new Image;
img.onload = () => {
var canvas = document.getElementById('canvas');
canvas.width = img.width;
canvas.height = img.height;
var ctx = canvas.getContext('2d');
ctx.drawImage(img, 0, 0);
document.getElementById('button').addEventListener('click', ()=>{
autoCropCanvas(canvas, ctx);
document.getElementById('button').remove();
});
};
img.src = '';
function autoCropCanvas(canvas, ctx) {
var bounds = {
left: 0,
right: canvas.width,
top: 0,
bottom: canvas.height
};
var rows = [];
var cols = [];
var imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);
for (var x = 0; x < canvas.width; x++) {
cols[x] = cols[x] || false;
for (var y = 0; y < canvas.height; y++) {
rows[y] = rows[y] || false;
const p = y * (canvas.width * 4) + x * 4;
const [r, g, b, a] = [imageData.data[p], imageData.data[p + 1], imageData.data[p + 2], imageData.data[p + 3]];
var isEmptyPixel = Math.max(r, g, b, a) === 0;
if (!isEmptyPixel) {
cols[x] = true;
rows[y] = true;
}
}
}
for (var i = 0; i < rows.length; i++) {
if (rows[i]) {
bounds.top = i ? i - 1 : i;
break;
}
}
for (var i = rows.length; i--; ) {
if (rows[i]) {
bounds.bottom = i < canvas.height ? i + 1 : i;
break;
}
}
for (var i = 0; i < cols.length; i++) {
if (cols[i]) {
bounds.left = i ? i - 1 : i;
break;
}
}
for (var i = cols.length; i--; ) {
if (cols[i]) {
bounds.right = i < canvas.width ? i + 1 : i;
break;
}
}
var newWidth = bounds.right - bounds.left;
var newHeight = bounds.bottom - bounds.top;
var cut = ctx.getImageData(bounds.left, bounds.top, newWidth, newHeight);
canvas.width = newWidth;
canvas.height = newHeight;
ctx.putImageData(cut, 0, 0);
}
<canvas id=canvas style='border: 1px solid pink'></canvas>
<button id=button>crop canvas</button>