简体   繁体   English

从base64 PNG获取高度和宽度尺寸

[英]Get height and width dimensions from base64 PNG

I have this我有这个



I would like to get the height and width from this string using JavaScript.我想使用 JavaScript 从这个字符串中获取高度和宽度。 How would I do this?我该怎么做? Is it even possible?甚至有可能吗?

You can assume access to jQuery, window.btoa, and window.atob.您可以假设访问 jQuery、window.btoa 和 window.atob。

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):我确信可以以某种方式从 PNG 中解析出它,但假设数据 URI 支持(因为我们可以假设atob ),您可以创建一个图像并等待它加载(这适用于任何格式):

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. PNG 文件以字节89 50 4E 47 0D 01 1A 0A开头,后跟包含宽度和高度且必须是第一个块的 IHDR 块。 (Yay, easier!) A chunk has a 4-byte length, a 4-byte type, and then a length -byte content. (是的,更简单!)一个块有一个 4 字节的长度,一个 4 字节的类型,然后是一个长度字节的内容。 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! IHDR 的内容以 4 字节的宽度和 4 字节的高度开始,因此 PNG 的宽度和高度始终是 16-24 字节! This can all be checked if you like, but for a simple way that assumes the PNG is valid:如果您愿意,可以检查所有这些,但对于假设 PNG 有效的简单方法:

 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);

And here is a new updated es6 version 3 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.这是 3 年后更新的新 es6 版本,在同步类型化数组的帮助下,您无需将孔图像加载到内存中即可弄清楚。 So its faster :)所以它更快:)

also don't require any DOM so it can work inside Workers也不需要任何 DOM,因此它可以在 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)


edit: 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 use more memory.编辑:我的建议也是你应该尽可能使用类型化数组而不是 base64 和 blob 而不是类型化数组,base64 是更糟糕的容器并使用更多内存。

So here is a solution for you who already have a blob:因此,对于已经有 blob 的您,这里有一个解决方案:

 document.createElement('canvas').toBlob(async blob => { // blob.arrayBuffer() is the new way to read stuff // may not work in all browser 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's // could expect this is more havier/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 }) })

LIVE DEMO现场演示

var imgData = '.........';    
var img = new Image();       
img.onload = function(){
  alert(img.width +' '+ img.height );
};
img.src = imgData;

With jQuery:使用 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)

** in node.js ** 在 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)

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM