18

I have this:



I would like to get the height and width from this string using JavaScript. How would I do this? Is it even possible?

You can assume access to jQuery, window.btoa, and window.atob.

Peter Mortensen
  • 30,738
  • 21
  • 105
  • 131
Travis Smith
  • 538
  • 2
  • 9
  • 21
  • 5
    Put it in an `Image` object. – SLaks Mar 10 '13 at 21:36
  • Specifically, I am looking for something that will parse and read the base64 string and do two things: (1) Verify that it is a PNG (Note: to me, it appears that all PNGs have the initial substring 'iVBORw0KGgoAAAANSUhEUgAAA'), & (2) Determine from the IHDR header the width, height (& it would be great to get the rest as well: bit depth, color type, compression mode, filter, and interlace). If possible, I'd like to avoid canvas or Image. – Travis Smith Mar 11 '13 at 17:02
  • It is not very likely that you will be able to write a PNG parser in Javascript that is faster than the browser's native parser. – SLaks Mar 11 '13 at 17:27
  • I don't believe it is possible to get this information with javascript/jQuery, since this information is simply handled by the browser and not exposed to javascript in any way I know of. It makes me wonder where the data uri comes from, and whether you can't get the information before it arrives in the webpage. – Sygmoral Mar 11 '13 at 17:28
  • 3
    @Sygmoral: It is certainly possible to parse the PNG file format in pure JS. – SLaks Mar 11 '13 at 17:31

6 Answers6

41

I’m sure that it’s possible to parse that out of the PNG somehow, but assuming data URI support (since we can assume atob), you can just create an image and wait for it to load (this works in any format):

var image = document.createElement('img');

image.addEventListener('load', function() {
    // image.width × image.height
});

image.src = 'data:image/png;base64,…';

Here's a demo.


Okay, apparently you’d like to extract this information manually. A PNG file starts with the bytes 89 50 4E 47 0D 01 1A 0A, followed by the IHDR chunk that contains the width and height and must be the first chunk. (Yay, easier!) A chunk has a 4-byte length, a 4-byte type, and then a length-byte content. IHDR’s content starts with a 4-byte width and a 4-byte height, so a PNG’s width and height are always bytes 16–24! This can all be checked if you like, but for a simple way that assumes the PNG is valid:

function toInt32(bytes) {
    return (bytes[0] << 24) | (bytes[1] << 16) | (bytes[2] << 8) | bytes[3];
}

function getDimensions(data) {
    return {
        width: toInt32(data.slice(16, 20)),
        height: toInt32(data.slice(20, 24))
    };
}

var base64Characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';

function base64Decode(data) {
    var result = [];
    var current = 0;

    for(var i = 0, c; c = data.charAt(i); i++) {
        if(c === '=') {
            if(i !== data.length - 1 && (i !== data.length - 2 || data.charAt(i + 1) !== '=')) {
                throw new SyntaxError('Unexpected padding character.');
            }

            break;
        }

        var index = base64Characters.indexOf(c);

        if(index === -1) {
            throw new SyntaxError('Invalid Base64 character.');
        }

        current = (current << 6) | index;

        if(i % 4 === 3) {
            result.push(current >> 16, (current & 0xff00) >> 8, current & 0xff);
            current = 0;
        }
    }

    if(i % 4 === 1) {
        throw new SyntaxError('Invalid length for a Base64 string.');
    }

    if(i % 4 === 2) {
        result.push(current >> 4);
    } else if(i % 4 === 3) {
        current <<= 6;
        result.push(current >> 16, (current & 0xff00) >> 8);
    }

    return result;
}

function getPngDimensions(dataUri) {
    if (dataUri.substring(0, 22) !== 'data:image/png;base64,') {
        throw new Error('Unsupported data URI format');
    }

    // 32 base64 characters encode the necessary 24 bytes
    return getDimensions(base64Decode(dataUri.substr(22, 32)));
}

var dimensions = getPngDimensions('');

console.log(dimensions.width + ' × ' + dimensions.height);
Ry-
  • 218,210
  • 55
  • 464
  • 476
  • 3
    I would have gone with `new Image()` – Musa Mar 10 '13 at 21:41
  • This works great! So I apologize for being unclear in my question. See new comment above. – Travis Smith Mar 11 '13 at 17:02
  • @TravisSmith: Okay :D After much arduous toil, difficult labour and hard work, I read a paragraph of specification and made a thing for that. You might want to add some error-checking, though. – Ry- Mar 11 '13 at 20:37
  • 1
    For those banging their heads on the shortcut via DOM, the IMGs created via `new Image()` or `document.createElement("IMG")` won't have dimensions until _after_ it loads. I figured BASE64 images would "load" instantly... **WRONG!** Adding the "load" event listener and _then_ acting on the height and width was the key. Shameful oversight on my part. :( – Eric L. Jun 24 '13 at 19:37
  • 1
    [As per the spec](https://developer.mozilla.org/en-US/docs/Web/API/HTMLImageElement), there are also the `naturalHeight` and `naturalWidth` properties, which may be more appropriate in cases where you actually display and style the . – Deiwin Mar 24 '15 at 11:59
  • @Deiwin yours should be an answer if not **the** answer – tbm0115 Jun 26 '17 at 22:34
  • a bit of a waste to parse the hole base64 data when you could just read a pice of chunk of it – Endless May 12 '20 at 20:15
  • @Endless: True! Fixed. – Ry- May 12 '20 at 23:40
21

And here is a new updated ES6 version three years later with a tiny bit of help from typed arrays that is synchronous and where you don't have to load the hole image to memory to figure it out. So it’s faster :)

Also, it doesn't require any DOM, so it can work inside Workers.

function getPngDimensions(base64) {
  const header = atob(base64.slice(0, 50)).slice(16,24)
  const uint8 = Uint8Array.from(header, c => c.charCodeAt(0))
  const dataView = new DataView(uint8.buffer)

  return {
    width: dataView.getInt32(0),
    height: dataView.getInt32(4)
  }
}

// Just to get some Base64 PNG example
const canvas = document.createElement('canvas')
const base64 = canvas.toDataURL().split(',')[1]

const dimensions = getPngDimensions(base64)
console.log(dimensions)

My recommendation is also that you should try to use typed arrays instead of Base64 and blobs instead of typed arrays when possible. Base64 is the worse container and uses more memory.

So here is a solution for you who already have a blob:

document.createElement('canvas').toBlob(async blob => {
  // blob.arrayBuffer() is the new way to read stuff
  // It may not work in all browsers
  let dv = new DataView(await blob.slice(16, 24).arrayBuffer())

  console.log({
    width: dv.getInt32(0),
    height: dv.getInt32(4)
  })
})

// You could also try out the new experimental createImageBitmap.
// Don't use image or canvas, but this also works in web workers.
// And it also works for more than just PNG images.
// We could expect this is more heavier/slower than just reading bytes 16-24.
document.createElement('canvas').toBlob(async blob => {
  const bitmap = await createImageBitmap(blob)
  const { width, height } = bitmap
  bitmap.close() // GC
  console.log({ width, height })
})
Peter Mortensen
  • 30,738
  • 21
  • 105
  • 131
Endless
  • 34,080
  • 13
  • 108
  • 131
  • 1
    This is great! Any chance you could help with a JPEG version of this function? The Specifications are over my head! – case2000 Jan 17 '17 at 15:39
  • I get the height in negative **{width: 4718592 height: -1962920}**. Resolution of the image if 2400x2400 – Musab Akram Dec 25 '19 at 06:12
  • hmm. just tried changing the canvas to 2400x2400, works just fine. did you try it on anything other than png maybe? (this just works for png). – Endless Dec 25 '19 at 16:40
  • Not working geting this errors: "Uncaught DOMException: Failed to execute 'atob' on 'Window': The string to be decoded is not correctly encoded" – Siraj Ali Nov 01 '22 at 13:37
  • @SirajAli Are you working with a PNG? if atob don't work, then try any of the other methods i provided. try to get a arrayBuffer and read Int32 at pos 16 & 24 – Endless Nov 02 '22 at 11:38
7

LIVE DEMO

var imgData = '.........';    
var img = new Image();       
img.onload = function(){
  alert(img.width +' '+ img.height );
};
img.src = imgData;
Roko C. Buljan
  • 196,159
  • 39
  • 305
  • 313
5

With jQuery:

var width, height;
$('<img/>').load(function(){
    width = $(this).width();
    height = $(this).height();
    // Call whatever function here that requires the width/height
}).attr('src', datauri)
Sygmoral
  • 7,021
  • 2
  • 23
  • 31
3

In Node.js:

function getPNGSize (buffer) {
  if (buffer.toString('ascii', 12, 16) === 'CgBI') {
    return {
      'width': buffer.readUInt32BE(32),
      'height': buffer.readUInt32BE(36)
    };
  }
  return {
    'width': buffer.readUInt32BE(16),
    'height': buffer.readUInt32BE(20)
  };
}
getPNGSize(buffer)
Peter Mortensen
  • 30,738
  • 21
  • 105
  • 131
аlex
  • 5,426
  • 1
  • 29
  • 38
0

You could simply get away with using Image. This returns a tuple of [<width>, <height>].

async function getBase64Dimensions(src) {
  return new Promise((resolve, reject) => {
    const image = new Image();

    image.src = src;

    image.onload = () => resolve([image.width, image.height]);
    image.onerror = () => reject();
  });
}

Type annotations if you can't live without them like me:

async function getBase64Dimensions(src: string) {
  return new Promise<[number, number]>((resolve, reject) => {
    const image = new Image();

    image.src = src;

    image.onload = () => resolve([image.width, image.height]);
    image.onerror = () => reject();
  });
}
TrèsAbhi
  • 68
  • 1
  • 9