Как преобразовать PNG Base 64 в пиксельный массив без использования image.src или image.onload? - PullRequest
1 голос
/ 28 июня 2019

Мне было интересно, как загрузить base64 на сайты, на которых есть CSP, которые этого не позволяют. Это означает, что мы не можем делать такие вещи, как:

image.src = "data:image/png;base64..."

Так что я решил, что мне нужно найти другой подход. Я попытался преобразовать base64 в двоичную строку, а затем поработать с ней, чтобы получить UintArray, содержащий данные изображения пикселей.

Я не знаю, как получить и получить изображение из функции loadImage , и при этом я не знаю, что я делаю неправильно, когда получаю пиксель (-ы).

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

Вот мой код, который пытается решить эту проблему:

        var canvas = document.getElementById("canvas");

        var imageBase64 = "";

        canvas.width = 540;
        canvas.height = 360;

        var ctx = canvas.getContext('2d');

        function loadImage(base64)
        {
            var binary_string = window.atob(base64.split(",")[1]);
            var len = binary_string.length;
            var pixels = new Uint8Array(len);

            for (var i = 0; i < len; i++)        
            {
                pixels[i] = binary_string.charCodeAt(i);
            }

            putImageData(ctx, {
                data: pixels,
                width: 100,
                height: 100
            }, 0, 0);



            var img = new Image();
            img.crossOrigin = "Anonymous"; 

            console.log(pixels);

            return img;
        }

        loadImage(imageBase64);

        //ctx.drawImage(loadImage(imageBase64), 0, 0);



        function putImageData(ctx, imageData, dx, dy, dirtyX, dirtyY, dirtyWidth, dirtyHeight) 
        {
            var data = imageData.data;
            var height = imageData.height;
            var width = imageData.width;

            dirtyX = dirtyX || 0;
            dirtyY = dirtyY || 0;
            dirtyWidth = dirtyWidth !== undefined ? dirtyWidth : width;
            dirtyHeight = dirtyHeight !== undefined ? dirtyHeight : height;

            var limitBottom = dirtyY + dirtyHeight;
            var limitRight = dirtyX + dirtyWidth;

            for (var y = dirtyY; y < limitBottom; y++) 
            {
                for (var x = dirtyX; x < limitRight; x++) 
                {
                    var pos = y * width + x;
                    ctx.fillStyle = 'rgba(' + data[pos * 4 + 0]
                                    + ',' + data[pos * 4 + 1]
                                    + ',' + data[pos * 4 + 2]
                                    + ',' + (data[pos * 4 + 3] / 255) + ')';
                    ctx.fillRect(x + dx, y + dy, 1, 1);
                }
            }
        }

Может быть, если мы сможем выяснить, как работает base64, мы сможем это сделать. И с этим: Любая помощь будет оценена :) Спасибо!

Ответы [ 2 ]

3 голосов
/ 28 июня 2019

В отличие от ожидаемого, двоичные данные, представляющие изображение PNG, не являются массивом пикселей.Двоичные данные имеют заголовки и различные метаданные, описывающие файл, а затем сжатое изображение.

Если вам нужно извлечь пиксели из двоичного файла png без использования встроенной функции декодирования браузера, вам нужно либо изучить PNGотформатируйте себя или используйте специализированную библиотеку, которая предоставит вам доступ к пикселям изображения.

Существует множество библиотек изображений, которые поддерживают формат PNG.Ниже приводится специализированная библиотека, которая может помочь вам.

https://www.npmjs.com/package/pngjs

Вы можете использовать Browserify , чтобы использовать эту библиотеку в клиентской части.

1 голос
/ 28 июня 2019

Код для просмотра здесь сгенерированный файл (480 КБ) :

var PNG = require('pngjs').PNG;

function getPNG(bin) {
    return PNG.sync.read(bin);
}

function loadImage(base64)
{
    var binary_string = Buffer.from(base64, 'base64');

    var png = getPNG(binary_string);

    return png;
}

window.getPNG = getPNG;
window.loadImage = loadImage;

И фрагмент кода:

<canvas id=canvas></canvas>
<script src="https://pastebin.com/raw/SN2vwZeg"></script>
<script>
var canvas = document.getElementById("canvas");

var imageBase64 = ""

var ctx = canvas.getContext('2d');

var png = loadImage(imageBase64.split(",")[1]);

putImageData(ctx, {
        data: png.data,
        width: png.width,
        height: png.height
    }, 0, 0);


function putImageData(ctx, imageData, dx, dy, dirtyX, dirtyY, dirtyWidth, dirtyHeight) 
{
    var data = imageData.data;
    var height = imageData.height;
    var width = imageData.width;

    dirtyX = dirtyX || 0;
    dirtyY = dirtyY || 0;
    dirtyWidth = dirtyWidth !== undefined ? dirtyWidth : width;
    dirtyHeight = dirtyHeight !== undefined ? dirtyHeight : height;

    var limitBottom = dirtyY + dirtyHeight;
    var limitRight = dirtyX + dirtyWidth;

    for (var y = dirtyY; y < limitBottom; y++) 
    {
        for (var x = dirtyX; x < limitRight; x++) 
        {
            var pos = y * width + x;
            ctx.fillStyle = 'rgba(' + data[pos * 4 + 0]
                            + ',' + data[pos * 4 + 1]
                            + ',' + data[pos * 4 + 2]
                            + ',' + (data[pos * 4 + 3] / 255) + ')';
            ctx.fillRect(x + dx, y + dy, 1, 1);
        }
    }
}
</script>
...