aboutsummaryrefslogtreecommitdiffstats
path: root/web_src/js/utils/image.ts
diff options
context:
space:
mode:
Diffstat (limited to 'web_src/js/utils/image.ts')
-rw-r--r--web_src/js/utils/image.ts48
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};
+}