mirror of
https://github.com/go-gitea/gitea.git
synced 2025-08-10 02:34:04 +09:00
1. change function comments to the minimal [tsdoc](https://tsdoc.org/) style. This has the benefit of making editors show the doc string in tooltips when the function is hovered: <img width="521" height="110" alt="image" src="https://github.com/user-attachments/assets/b966f4f1-8239-433a-a456-5bd5c05d69ef" /> 2. disable eslint `multiline-comment-style` as it conflicts with tsdoc. --------- Signed-off-by: silverwind <me@silverwind.io>
59 lines
1.7 KiB
TypeScript
59 lines
1.7 KiB
TypeScript
type PngChunk = {
|
|
name: string,
|
|
data: Uint8Array,
|
|
}
|
|
|
|
export async function pngChunks(blob: Blob): Promise<PngChunk[]> {
|
|
const uint8arr = new Uint8Array(await blob.arrayBuffer());
|
|
const chunks: PngChunk[] = [];
|
|
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;
|
|
}
|
|
|
|
type ImageInfo = {
|
|
width?: number,
|
|
dppx?: number,
|
|
}
|
|
|
|
/** decode a image and try to obtain width and dppx. It will never throw but instead
|
|
* return default values. */
|
|
export async function imageInfo(blob: Blob): Promise<ImageInfo> {
|
|
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};
|
|
}
|