diff options
Diffstat (limited to 'web_src/js/utils/image.ts')
-rw-r--r-- | web_src/js/utils/image.ts | 48 |
1 files changed, 48 insertions, 0 deletions
diff --git a/web_src/js/utils/image.ts b/web_src/js/utils/image.ts new file mode 100644 index 0000000000..c71d715941 --- /dev/null +++ b/web_src/js/utils/image.ts @@ -0,0 +1,48 @@ +export async function pngChunks(blob) { + const uint8arr = new Uint8Array(await blob.arrayBuffer()); + const chunks = []; + if (uint8arr.length < 12) return chunks; + const view = new DataView(uint8arr.buffer); + if (view.getBigUint64(0) !== 9894494448401390090n) return chunks; + + const decoder = new TextDecoder(); + let index = 8; + while (index < uint8arr.length) { + const len = view.getUint32(index); + chunks.push({ + name: decoder.decode(uint8arr.slice(index + 4, index + 8)), + data: uint8arr.slice(index + 8, index + 8 + len), + }); + index += len + 12; + } + + return chunks; +} + +// decode a image and try to obtain width and dppx. It will never throw but instead +// return default values. +export async function imageInfo(blob) { + let width = 0, dppx = 1; // dppx: 1 dot per pixel for non-HiDPI screens + + if (blob.type === 'image/png') { // only png is supported currently + try { + for (const {name, data} of await pngChunks(blob)) { + const view = new DataView(data.buffer); + if (name === 'IHDR' && data?.length) { + // extract width from mandatory IHDR chunk + width = view.getUint32(0); + } else if (name === 'pHYs' && data?.length) { + // extract dppx from optional pHYs chunk, assuming pixels are square + const unit = view.getUint8(8); + if (unit === 1) { + dppx = Math.round(view.getUint32(0) / 39.3701) / 72; // meter to inch to dppx + } + } + } + } catch {} + } else { + return {}; // no image info for non-image files + } + + return {width, dppx}; +} |