/* Copyright 2021 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 { MatrixEvent } from "matrix-js-sdk/src/matrix"; import { EventType, MsgType } from "matrix-js-sdk/src/@types/event"; import { logger } from "matrix-js-sdk/src/logger"; import { LazyValue } from "./LazyValue"; import { Media, mediaFromContent } from "../customisations/Media"; import { decryptFile } from "./DecryptFile"; import { IMediaEventContent } from "../customisations/models/IMediaEventContent"; import { IDestroyable } from "./IDestroyable"; // TODO: We should consider caching the blobs. https://github.com/vector-im/element-web/issues/17192 export class MediaEventHelper implements IDestroyable { // Either an HTTP or Object URL (when encrypted) to the media. public readonly sourceUrl: LazyValue; public readonly thumbnailUrl: LazyValue; // Either the raw or decrypted (when encrypted) contents of the file. public readonly sourceBlob: LazyValue; public readonly thumbnailBlob: LazyValue; public readonly media: Media; public constructor(private event: MatrixEvent) { this.sourceUrl = new LazyValue(this.prepareSourceUrl); this.thumbnailUrl = new LazyValue(this.prepareThumbnailUrl); this.sourceBlob = new LazyValue(this.fetchSource); this.thumbnailBlob = new LazyValue(this.fetchThumbnail); this.media = mediaFromContent(this.event.getContent()); } public get fileName(): string { return ( this.event.getContent().filename || this.event.getContent().body || "download" ); } public destroy(): void { if (this.media.isEncrypted) { if (this.sourceUrl.cachedValue) URL.revokeObjectURL(this.sourceUrl.cachedValue); if (this.thumbnailUrl.cachedValue) URL.revokeObjectURL(this.thumbnailUrl.cachedValue); } } private prepareSourceUrl = async (): Promise => { if (this.media.isEncrypted) { const blob = await this.sourceBlob.value; return URL.createObjectURL(blob); } else { return this.media.srcHttp; } }; private prepareThumbnailUrl = async (): Promise => { if (this.media.isEncrypted) { const blob = await this.thumbnailBlob.value; if (blob === null) return null; return URL.createObjectURL(blob); } else { return this.media.thumbnailHttp; } }; private fetchSource = (): Promise => { if (this.media.isEncrypted) { const content = this.event.getContent(); return decryptFile(content.file!, content.info); } return this.media.downloadSource().then((r) => r.blob()); }; private fetchThumbnail = (): Promise => { if (!this.media.hasThumbnail) return Promise.resolve(null); if (this.media.isEncrypted) { const content = this.event.getContent(); if (content.info?.thumbnail_file) { return decryptFile(content.info.thumbnail_file, content.info.thumbnail_info); } else { // "Should never happen" logger.warn("Media claims to have thumbnail and is encrypted, but no thumbnail_file found"); return Promise.resolve(null); } } return fetch(this.media.thumbnailHttp).then((r) => r.blob()); }; public static isEligible(event: MatrixEvent): boolean { if (!event) return false; if (event.isRedacted()) return false; if (event.getType() === EventType.Sticker) return true; if (event.getType() !== EventType.RoomMessage) return false; const content = event.getContent(); const mediaMsgTypes: string[] = [MsgType.Video, MsgType.Audio, MsgType.Image, MsgType.File]; if (mediaMsgTypes.includes(content.msgtype!)) return true; if (typeof content.url === "string") return true; // Finally, it's probably not media return false; } }