Как найти координаты пустых областей на картинке через JS и подставить туда другие изображения?

142
18 ноября 2018, 00:20

Возникла следующая проблема:

У меня есть множество различных картинок подобного рода:

На них есть прозрачные области различной формы, в которые мне нужно как-то подставить другие изображения, подогнав их размеры под габариты этих регионов.

Для этого нужно как-то получить координаты этих самых прозрачных областей на картинке.

Подскажите, как можно было бы реализовать данную задумку, так как я никак не могу к ней подступиться...

Answer 1

Начать хочу с того, что катастрофически не согласен с мнением участника @qwabra, что часть вопроса о вставке изображений

выходит за рамки вопроса

Человек описал свою проблему - значит, поможем ее решить полностью)

Итак. С чего бы подступиться к Вашей проблеме?

Для начала определимся с тем, что же Вам нужно. А вам нужно найти такие прямоугольные области на изображении, в которые Вы бы могли вписать другие изображения.
Почему прямоугольные? Я вроде как ничего в этом мире еще не проспал и растровые изображения по сей день

представляются как прямоугольный двумерный массив чисел


Отлично. Мы определились с тем, что необходимо искать на изображении прямоугольную область для вставки. Но как же быть? Вот у Вас на картинке есть круги и прочие отличные от прямоугольников фигуры, куда надо вписать изображение...
Собственно, любую фигуру можно заключить в прямоугольник, для этого нам нужно лишь:

  1. Взять наименьший X из множества точек, принадлежащих фигуре (далее - множество)
  2. Взять наименьший Y из множества
  3. Взять наибольший X из множества
  4. Взять наибольший Y из множества
  5. Скомпоновать полученные значения в (Xmin, Ymin) и (Xmax, Ymax), что явится верхней левой и правой нижней точками искомого прямоугольника

Хорошо. А как же нам получить эти самые X и Y? Надо как-то обследовать изображение, выделив пустые регионы.

Небольшое отступление:
Вы написали про "белые области" в заголовке своего вопроса. Однако, исходя из данной Вами картинки, Вы явно имели в виду области, где альфа-канал пикселей равен 0 (то есть области прозрачны). Дальнейший алгоритм базируется на этом небольшом исправлении.

Если же Вам потребуются все таки белые области, то алгоритм можно будет, конечно, доработать, но совершенно непонятно, что же делать с белыми областями по бокам изображений...

Возвращаемся к нашим рассуждениям)

Как обнаружить пустые регионы? Я предлагаю Вам такой алгоритм:

  1. Проходимся по каждому пикселю изображения
  2. Если пиксель удовлетворяет заданному условию (в нашем случае: альфа-канал == 0), а также не содержится в массиве уже найденных регионов, начинаем рекурсивно обходить его соседей, "цепляясь" за те пиксели, которые удовлетворяют нашему селектору, при этом соседствуя с такими же. Читаем с них данные
  3. После завершения рекурсивного обхода, у нас на руках будут данные об искомых X и Y (min/max), а также о числе пикселей в регионе
  4. Если число пикселей больше некоторого порога (нас не интересуют области в 2-3 пикселя, правда? Что там рисовать? ¯\_(ツ)_/¯), то добавляем данные о пройденном регионе в массив

Обходить изображение мы будем от верхнего левого угла к правому нижнему.
В качестве оптимизации рекурсивного обхода могу предложить проводить его следующим образом:
Так как у Вас предполагаются простые фигуры для областей, куда могут быть вставлены изображения (безо всяких рюшечек посреди них), мы можем исследовать области построчно. Поясню:
Так как мы обходим изображение с самого верха, то и за искомую область мы "зацепимся" в высшей ее точке. Далее мы можем запустить функцию проверки по той же строчке влево и вправо, проверяя соседние пиксели на соответствие условию. Когда проверка заканчивается, мы спускаемся на пиксель ниже и проверяем следующую строку

Данный метод будет отрабатывать достаточно быстро, однако, подчеркну, он пригоден лишь для простых фигур! Если появятся витиеватости, то придется переписать рекурсивный обход

Итак, после обхода изображения у нас есть информация по всем пустым регионам на нем.
Немного спойлеров: пройдясь описанным мною методом по Вашему изображению, я получил следующие области:

(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>

Если Вы нажмете на кнопку в сниппете, то увидите, как Ваша картинка объединится с моими изображениями вот в это:

Думаю, именно этого результата Вы и хотели достичь!

Надеюсь, мой ответ смог Вам помочь и Вы продолжите двигаться к своим свершениями)
Если что-то было не очень понятно, либо же Вам требуется помощь в доработке описанного метода - не стесняйтесь и спрашивайте!

Answer 2
Как найти координаты белых или прозрачных областей на картинке с помощью JS?

В них должны вставляться другие картинки с автоизменением их размера

выходит за рамки вопроса

Для этого нужно как-то получить координаты этих белых областей на картинке

вот пример создания матрицы искомых областей 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

код на TypeScript с комментариями

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) // - белая область
        }
    })
}
READ ALSO
Uncaught SyntaxError: Unexpected token &lt; Что это за ошибка?

Uncaught SyntaxError: Unexpected token < Что это за ошибка?

Эта ошибка возникает, когда перемещаю код jquery (отправляет данные на сервер node js) в отделный файл

232
Вставка символа где-то в строку

Вставка символа где-то в строку

Допустим есть строка с временем, например 1234, по какой-то причине она не разделена, привычным нам, двоеточиемМы знаем, что двоеточие будет...

149
Как зашифровать сообщение в aes256 на чистом JavaScript?

Как зашифровать сообщение в aes256 на чистом JavaScript?

Охота зашифровать сообщение в AES256Допустим есть ключ и есть сообщение

149