/* Copyright 2022 The Matrix.org Foundation C.I.C. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ import { BlurhashEncoder } from "../BlurhashEncoder"; import { EncryptedFile } from "../customisations/models/IMediaEventContent"; type ThumbnailableElement = HTMLImageElement | HTMLVideoElement; export const BLURHASH_FIELD = "xyz.amorgan.blurhash"; // MSC2448 interface IThumbnail { info: { thumbnail_info?: { w: number; h: number; mimetype: string; size: number; }; w: number; h: number; [BLURHASH_FIELD]?: string; thumbnail_url?: string; thumbnail_file?: EncryptedFile; }; thumbnail: Blob; } const MAX_WIDTH = 800; const MAX_HEIGHT = 600; /** * Create a thumbnail for a image DOM element. * The image will be smaller than MAX_WIDTH and MAX_HEIGHT. * The thumbnail will have the same aspect ratio as the original. * Draws the element into a canvas using CanvasRenderingContext2D.drawImage * Then calls Canvas.toBlob to get a blob object for the image data. * * Since it needs to calculate the dimensions of the source image and the * thumbnailed image it returns an info object filled out with information * about the original image and the thumbnail. * * @param {HTMLElement} element The element to thumbnail. * @param {number} inputWidth The width of the image in the input element. * @param {number} inputHeight the width of the image in the input element. * @param {string} mimeType The mimeType to save the blob as. * @param {boolean} calculateBlurhash Whether to calculate a blurhash of the given image too. * @return {Promise} A promise that resolves with an object with an info key * and a thumbnail key. */ export async function createThumbnail( element: ThumbnailableElement, inputWidth: number, inputHeight: number, mimeType: string, calculateBlurhash = true, ): Promise { let targetWidth = inputWidth; let targetHeight = inputHeight; if (targetHeight > MAX_HEIGHT) { targetWidth = Math.floor(targetWidth * (MAX_HEIGHT / targetHeight)); targetHeight = MAX_HEIGHT; } if (targetWidth > MAX_WIDTH) { targetHeight = Math.floor(targetHeight * (MAX_WIDTH / targetWidth)); targetWidth = MAX_WIDTH; } let canvas: HTMLCanvasElement | OffscreenCanvas; let context: CanvasRenderingContext2D | OffscreenCanvasRenderingContext2D; try { canvas = new window.OffscreenCanvas(targetWidth, targetHeight); context = canvas.getContext("2d") as OffscreenCanvasRenderingContext2D; } catch (e) { // Fallback support for other browsers (Safari and Firefox for now) canvas = document.createElement("canvas"); canvas.width = targetWidth; canvas.height = targetHeight; context = canvas.getContext("2d")!; } context.drawImage(element, 0, 0, targetWidth, targetHeight); let thumbnailPromise: Promise; if (window.OffscreenCanvas && canvas instanceof OffscreenCanvas) { thumbnailPromise = canvas.convertToBlob({ type: mimeType }); } else { thumbnailPromise = new Promise((resolve) => (canvas as HTMLCanvasElement).toBlob(resolve as BlobCallback, mimeType), ); } const imageData = context.getImageData(0, 0, targetWidth, targetHeight); // thumbnailPromise and blurhash promise are being awaited concurrently const blurhash = calculateBlurhash ? await BlurhashEncoder.instance.getBlurhash(imageData) : undefined; const thumbnail = await thumbnailPromise; return { info: { thumbnail_info: { w: targetWidth, h: targetHeight, mimetype: thumbnail.type, size: thumbnail.size, }, w: inputWidth, h: inputHeight, [BLURHASH_FIELD]: blurhash, }, thumbnail, }; }