mirror of
				https://github.com/go-gitea/gitea
				synced 2025-10-26 08:58:24 +00: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};
 | |
| }
 |