Возникла следующая проблема:
У меня есть множество различных картинок подобного рода:
На них есть прозрачные области различной формы, в которые мне нужно как-то подставить другие изображения, подогнав их размеры под габариты этих регионов.
Для этого нужно как-то получить координаты этих самых прозрачных областей на картинке.
Подскажите, как можно было бы реализовать данную задумку, так как я никак не могу к ней подступиться...
Начать хочу с того, что катастрофически не согласен с мнением участника @qwabra, что часть вопроса о вставке изображений
выходит за рамки вопроса
Человек описал свою проблему - значит, поможем ее решить полностью)
Итак. С чего бы подступиться к Вашей проблеме?
Для начала определимся с тем, что же Вам нужно. А вам нужно найти такие прямоугольные области на изображении, в которые Вы бы могли вписать другие изображения.
Почему прямоугольные? Я вроде как ничего в этом мире еще не проспал и растровые изображения по сей день
представляются как прямоугольный двумерный массив чисел
Отлично. Мы определились с тем, что необходимо искать на изображении прямоугольную область для вставки. Но как же быть? Вот у Вас на картинке есть круги и прочие отличные от прямоугольников фигуры, куда надо вписать изображение...
Собственно, любую фигуру можно заключить в прямоугольник, для этого нам нужно лишь:
Хорошо. А как же нам получить эти самые X и Y? Надо как-то обследовать изображение, выделив пустые регионы.
Небольшое отступление:
Вы написали про "белые области" в заголовке своего вопроса. Однако, исходя из данной Вами картинки, Вы явно имели в виду области, где альфа-канал пикселей равен 0 (то есть области прозрачны). Дальнейший алгоритм базируется на этом небольшом исправлении.
Если же Вам потребуются все таки белые области, то алгоритм можно будет, конечно, доработать, но совершенно непонятно, что же делать с белыми областями по бокам изображений...
Возвращаемся к нашим рассуждениям)
Как обнаружить пустые регионы? Я предлагаю Вам такой алгоритм:
Обходить изображение мы будем от верхнего левого угла к правому нижнему.
В качестве оптимизации рекурсивного обхода могу предложить проводить его следующим образом:
Так как у Вас предполагаются простые фигуры для областей, куда могут быть вставлены изображения (безо всяких рюшечек посреди них), мы можем исследовать области построчно. Поясню:
Так как мы обходим изображение с самого верха, то и за искомую область мы "зацепимся" в высшей ее точке. Далее мы можем запустить функцию проверки по той же строчке влево и вправо, проверяя соседние пиксели на соответствие условию. Когда проверка заканчивается, мы спускаемся на пиксель ниже и проверяем следующую строку
Данный метод будет отрабатывать достаточно быстро, однако, подчеркну, он пригоден лишь для простых фигур! Если появятся витиеватости, то придется переписать рекурсивный обход
Итак, после обхода изображения у нас есть информация по всем пустым регионам на нем.
Немного спойлеров: пройдясь описанным мною методом по Вашему изображению, я получил следующие области:
(29, 24) - (159, 143)
(183, 26) - (304, 147)
(369, 31) - (625, 187)
(81, 156) - (260, 297)
(369, 274) - (625, 431)
(181, 311) - (302, 431)
(29, 312) - (165, 433)
Или, если визуализировать:
Что же теперь?
А теперь дело за малым!
Еще раз обращу Ваше внимание на то, что изображения у Вас содержат прозрачные области.
В переложении на реальную жизнь: если бы Вы держали данную картинку в руках, в ней были бы дыры. Дыры, которыми очень удобно наложить это самое изображение на те фотографии, что должны быть на их месте
Надеюсь, Вы уловили, к чему я веду
Мы можем отрисовать нужные нам изображения в заранее рассчитанных областях, а потом просто как бы положить сверху "дырявую" картинку, так что наши будут видны из-под нее
Фух. Наговорил я много, но также могу подкрепить свои слова кодом с комментариями)
Приведу его сразу в виде сниппета, чтобы Вы могли узреть результат (разверните на всю страницу):
// Наш обработчик изображений
function ImageExplorer(img, selector) {
// Var
ImageExplorer.minRegion = 5000;
let it = this;
let image;
let canvas;
let context;
let emptyRegions = [];
// Initialization
function init() {
image = getHTMLImageElement(img);
// Создаем канвас для работы, получаем его контекст
canvas = document.createElement("canvas");
context = canvas.getContext("2d");
// Проверяем переданный селектор, определяющий, подходит ли нам пиксель
if (selector == undefined)
// Альфа-канал равен 0 => пиксель прозрачен
selector = color => color[0] == 0;
}
// Funcs
// Ищем пустые регионы
it.findRegions = () => new Promise(resolve => {
// Ожидаем загрузки изображения
new Promise(res => {
let intervalID = setInterval(() => {
if (image.complete)
{
clearInterval(intervalID);
clearCanvas();
// Разрешаем текущее обещание
res();
}
}, 300);
}).then(() => {
emptyRegions = [];
// Отрисовываем картинку
context.drawImage(image, 0, 0);
// Получаем данные о ее пикселях
let imgData = context.getImageData(0, 0, image.width, image.height).data;
// Функция, которая по значениям x и y вернет цвет указанного пикселя ([a, r, g, b])
let getPixel = (dX, dY) => {
let index = 4 * (dX + dY * image.width);
return [imgData[index], imgData[index + 1], imgData[index + 2], imgData[index + 3]];
};
// Проходимся по каждому пикселю изображения
for (let y = 0; y < image.height; y++)
for (let x = 0; x < image.width; x++) {
// Проверяем, существует ли пустой регион, которому принадлежит текущий пиксель
let existing = emptyRegions.find(reg => reg.contains(x, y));
// Если нет, проверяем дальше
if (!existing) {
// Если пиксель подходит нам, начинаем обследование региона
if (selector(getPixel(x, y))) {
// Устанавливаем координаты для крайней левой верхней и крайней правой нижней точек
// Ниже они будут меняться
let left = x;
let up = y;
let right = x;
let down = y;
// Число пикселей в исследованном регионе
let pixels = 1;
// Самый сок
// Самовызывающаяся рекурсивная функция, ссылающаяся на текущий контекст
// Один из немногих поводов любить JS хД
//
// Логика такая: мы вызываем функцию на текущих координатах, а далее она
// "расползается" по всему региону, где соседствующие пиксели
// удовлетворяют селектору
(function(dX, dY, type) {
// Если мы не вышли за пределы изображения, а также пиксель подходит под селектор
if (dX > -1 && dX < image.width && dY < image.height && selector(getPixel(dX, dY))) {
// Заменяем значения, если требуется
if (dX < left)
left = dX;
if (dX > right)
right = dX;
if (dY < up)
up = dY;
if (dY > down)
down = dY;
// Увеличиваем число найденых пикселей в регионе
++pixels;
// type несет следующий смысл:
// 0: текущая функция запускает "исследование" по оси Ox, а потом переходит на пиксель ниже
// 1: текущая функция продолжает "исследование" по оси Ox (в сторону увеличения)
// 2: текущая функция продолжает "исследование" по оси Ox (в сторону убывания)
switch (type) {
case 0:
arguments.callee(dX + 1, dY, 1);
arguments.callee(dX - 1, dY, 2);
arguments.callee(dX, dY + 1, 0);
break;
case 1:
arguments.callee(dX + 1, dY, 1);
break;
case 2:
arguments.callee(dX - 1, dY, 2);
break;
default:
break;
}
}
})(x, y, 0);
// Если пикселей в регионе >= минимального кол-ва пикселей, то добавим этот регион
if (pixels >= ImageExplorer.minRegion)
{
emptyRegions.push(new EmptyRegion([left, up], [right, down]));
//x = right + 1; // Небольшая оптимизация. Дает не слишком большой выигрыш, так что выключена для большей точности
}
}
}
//else
//x = existing.right + 1; // Небольшая оптимизация. Дает не слишком большой выигрыш, так что выключена для большей точности
}
// Разрешаем текущее обещание, отдаем найденные регионы
resolve(emptyRegions);
});
});
// Создадим новое изображение, передав только изображения, которые нужно
// "подложить" под основную картинку (в указанном порядке)
it.createImage = function() {
clearCanvas();
// Преобразуем переданные аргументы в массив изображений
arguments[Symbol.isConcatSpreadable] = true; // https://ru.stackoverflow.com/a/860866/248572
let images = [].concat(arguments).map(getHTMLImageElement);
// Отрисуем каждое изображение, отмасштабировав его по сопутствующему региону
for (let i in images)
context.drawImage(images[i], emptyRegions[i].left, emptyRegions[i].up, emptyRegions[i].getWidth(), emptyRegions[i].getHeight());
// Положим основное изображение поверх
context.drawImage(image, 0, 0);
// Вернем результирующее изображение
let resultImage = new Image();
resultImage.src = canvas.toDataURL();
return resultImage;
};
// Создадим новое изображение, явно задав регионы
// Изображения будут отсортированы по наибольшему размерному соответствию переданным регионам
it.createImageFromRegions = function(regions, ...images) {
clearCanvas();
let regs = regions.slice();
// Преобразовываем входные аргументы в [изображение, наиболее подходящий регион]
let imgReg = images.map(x => {
// Получаем изображение
let result = getHTMLImageElement(x);
// Получаем соотношение его сторон
let ratio = result.width / result.height;
// Сортируем регионы по этому соотношению
regions = regs.sort((a, b) => a == b ? 0 : ((Math.abs(ratio - a.getRatio()) < Math.abs(ratio - b.getRatio())) ? -1 : 1));
// Нулевой регион будет наиболее подходящим
let bestRegion = regs[0];
// Удаляем его из массива
regs.splice(0, 1);
return [result, bestRegion];
});
// Отрисуем каждое изображение, отмасштабировав его по сопутствующему региону
for (let i in imgReg)
context.drawImage(imgReg[i][0], imgReg[i][1].left, imgReg[i][1].up, imgReg[i][1].getWidth(), imgReg[i][1].getHeight());
// Положим основное изображение поверх
context.drawImage(image, 0, 0);
// Вернем результирующее изображение
let resultImage = new Image();
resultImage.src = canvas.toDataURL();
return resultImage;
};
// Загружаем изображение
function getHTMLImageElement(data) {
let result;
if ((typeof data) == "string") {
result = new Image();
// Передана ссылка/Base64-строка
result.src = data;
}
else
if (data instanceof HTMLImageElement)
// Передан элемент <img>
result = data;
else
if (data[0] instanceof HTMLImageElement)
// Передан элемент <img>, бережно укутанный селектором jQuery
result = data[0];
else
throw "Необходимо передать изображение!"
return result;
}
// Растягиваем канвас до размеров изображения и чистим его
function clearCanvas() {
canvas.width = image.width;
canvas.height = image.height;
context.clearRect(0, 0, canvas.width, canvas.height);
}
init();
}
// Пустой регион
function EmptyRegion(LU, RD) {
// Var
// Добавочное значение
EmptyRegion.delta = 3;
let it = this;
it.left;
it.right;
it.up;
it.down;
// Initialization
function init(args) {
switch (args.length) {
case 2:
break;
case 4:
LU = [args[0], args[1]];
RD = [args[2], args[3]];
break;
default:
throw "Необходимо передать две точки!";
}
it.left = LU[0] - EmptyRegion.delta;
it.up = LU[1] - EmptyRegion.delta;
it.right = RD[0] + EmptyRegion.delta;
it.down = RD[1] + EmptyRegion.delta;
}
// Funcs
it.getWidth = () => it.right - it.left;
it.getHeight = () => it.down - it.up;
// Получим соотношение сторон региона
it.getRatio = () => it.getWidth() / it.getHeight();
// Содержит ли текущий регион указанную точку
it.contains = (x, y) => x >= it.left && x <= it.right && y >= it.up && y <= it.down;
init(arguments);
}
<script>
function getImgByID(id) {
return document.getElementById(id);
}
function create() {
let explorer = new ImageExplorer(getImgByID("main"));
explorer.findRegions().then(regions => {
let img = explorer.createImageFromRegions(regions, getImgByID("bill"),
getImgByID("ford"),
getImgByID("stan"),
getImgByID("dipper"),
getImgByID("mabel"),
getImgByID("soos"),
getImgByID("wendy"));
getImgByID("generated").src = img.src;
});
}
</script>
<h1>Главное изображение:</h1>
<img id="main">
<br><br><br>
<h1>Изображения для вставки:</h1>
<img id="bill">
<img id="ford">
<img id="stan">
<img id="dipper">
<img id="mabel">
<img id="soos">
<img id="wendy">
<br><br><br>
<h1>Скомпонованное изображение:</h1>
<img id="generated">
<br>
<button onclick="create()">Создать</button>
<script src="https://kir-antipov.github.io/StackOverflow/q872682.js"></script>
Если Вы нажмете на кнопку в сниппете, то увидите, как Ваша картинка объединится с моими изображениями вот в это:
Думаю, именно этого результата Вы и хотели достичь!
Надеюсь, мой ответ смог Вам помочь и Вы продолжите двигаться к своим свершениями)
Если что-то было не очень понятно, либо же Вам требуется помощь в доработке описанного метода - не стесняйтесь и спрашивайте!
В них должны вставляться другие картинки с автоизменением их размера
выходит за рамки вопроса
Для этого нужно как-то получить координаты этих белых областей на картинке
вот пример создания матрицы искомых областей arr[i][j] = ddq(i * mod, j * mod)
шаг поиска mod = 10
var lib;
(function (lib) {
lib.mod = 10;
})(lib || (lib = {}));
(function (lib) {
let addTobody = el => { document.body.appendChild(el); };
let callF = f => { f(); };
lib.load_elStack = [];
lib.load_fooStack = [];
window.addEventListener("load", () => {
lib.load_elStack.forEach(addTobody);
lib.load_fooStack.forEach(callF);
});
})(lib || (lib = {}));
(function (lib) {
lib.img = document.createElement('img');
lib.img.src = getImg();
})(lib || (lib = {}));
(function (lib) {
lib.canvas = document.createElement('canvas');
lib.load_elStack.push(lib.canvas);
lib.canvas.setAttribute('style', 'border: 1px solid grey;');
lib.ctx = lib.canvas.getContext("2d");
lib.load_fooStack.push(() => {
lib.canvas.width = lib.img.width;
lib.canvas.height = lib.img.height;
lib.ctx.drawImage(lib.img, 0, 0);
});
function ddq(x, y, q = 1) {
let p = lib.ctx.getImageData(x, y, 10, 10);
let d = p.data;
if (q) {
d = d.filter((q, i) => (i % 4) === 0);
d = d.filter(q => q);
}
else {
d = d.filter((q, i) => (i % 4) !== 0);
d = d.filter(q => q !== 255);
}
if (d.length === 0) {
lib.ctx.fillRect(x, y, lib.mod, lib.mod);
return 0;
}
return 1;
}
lib.load_fooStack.push(() => {
let arr = Array.from({ length: lib.canvas.width / lib.mod }, () => Array.from({ length: lib.canvas.height / lib.mod }));
if (true) {
for (var i = 0, l = arr.length; i < l; i++) {
for (var j = 0, l2 = arr[i].length; j < l2; j++) {
arr[i][j] = ddq(i * lib.mod, j * lib.mod);
}
}
}
else {
ddq(50, 50, 1);
ddq(2, 45, 0);
}
});
})(lib || (lib = {}));
function getImg() {
return '';
}
//# sourceMappingURL=index.js.map
namespace lib {
/**
* шаг - диаметр минимальной области поиска
*/
export const mod = 10
}
namespace lib {
let addTobody = el => { document.body.appendChild(el) }
let callF = f => { f() }
// --
export let load_elStack = [] as HTMLElement[]
export let load_fooStack = [] as Function[]
// --
window.addEventListener("load", () => {
load_elStack.forEach(addTobody)
load_fooStack.forEach(callF)
})
}
namespace lib {
export let img = document.createElement('img')
img.src = getImg()
// load_elStack.push(img)
}
namespace lib {
export let canvas = document.createElement('canvas')
load_elStack.push(canvas)
canvas.setAttribute('style', 'border: 1px solid grey;')
export let ctx = canvas.getContext("2d")
load_fooStack.push(() => {
canvas.width = img.width
canvas.height = img.height
ctx.drawImage(img, 0, 0)
})
function ddq(x, y, q = 1) {
let p = ctx.getImageData(x, y, 10, 10)
let d = p.data
if (q) {
// --> фильтер по нулефой прозрачности
d = d.filter((q, i) => (i % 4) === 0)
d = d.filter(q => q)
// <--
} else {
// --> фильтер по белым
d = d.filter((q, i) => (i % 4) !== 0)
d = d.filter(q => q !== 255)
// <--
}
// console.log(p)
// console.log(d.length)
// if (d.length !== 0) {
if (d.length === 0) {
ctx.fillRect(x, y, mod, mod)
return 0
}
return 1
}
load_fooStack.push(() => {
let arr = Array.from({ length: canvas.width / mod }, () => Array.from({ length: canvas.height / mod }))
if (true) {
for (var i = 0, l = arr.length; i < l; i++) {
for (var j = 0, l2 = arr[i].length; j < l2; j++) {
arr[i][j] = ddq(i * mod, j * mod)
}
}
} else {
ddq(50, 50, 1) // - прозрачная
ddq(2, 45, 0) // - белая область
}
})
}
Кофе для программистов: как напиток влияет на продуктивность кодеров?
Рекламные вывески: как привлечь внимание и увеличить продажи
Стратегії та тренди в SMM - Технології, що формують майбутнє сьогодні
Выделенный сервер, что это, для чего нужен и какие характеристики важны?
Современные решения для бизнеса: как облачные и виртуальные технологии меняют рынок
Эта ошибка возникает, когда перемещаю код jquery (отправляет данные на сервер node js) в отделный файл
Допустим есть строка с временем, например 1234, по какой-то причине она не разделена, привычным нам, двоеточиемМы знаем, что двоеточие будет...
Охота зашифровать сообщение в AES256Допустим есть ключ и есть сообщение