UNPKG

matrix-react-sdk

Version:
279 lines (273 loc) 42 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; var _extends2 = _interopRequireDefault(require("@babel/runtime/helpers/extends")); var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _react = _interopRequireDefault(require("react")); var _blurhash = require("blurhash"); var _logger = require("matrix-js-sdk/src/logger"); var _languageHandler = require("../../../languageHandler"); var _SettingsStore = _interopRequireDefault(require("../../../settings/SettingsStore")); var _InlineSpinner = _interopRequireDefault(require("../elements/InlineSpinner")); var _Media = require("../../../customisations/Media"); var _imageMedia = require("../../../utils/image-media"); var _MFileBody = _interopRequireDefault(require("./MFileBody")); var _ImageSize = require("../../../settings/enums/ImageSize"); var _RoomContext = _interopRequireWildcard(require("../../../contexts/RoomContext")); var _MediaProcessingError = _interopRequireDefault(require("./shared/MediaProcessingError")); function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); } function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; } /* Copyright 2024 New Vector Ltd. Copyright 2015-2021 The Matrix.org Foundation C.I.C. SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only Please see LICENSE files in the repository root for full details. */ class MVideoBody extends _react.default.PureComponent { constructor(...args) { super(...args); (0, _defineProperty2.default)(this, "videoRef", /*#__PURE__*/_react.default.createRef()); (0, _defineProperty2.default)(this, "sizeWatcher", void 0); (0, _defineProperty2.default)(this, "state", { fetchingData: false, decryptedUrl: null, decryptedThumbnailUrl: null, decryptedBlob: null, error: null, posterLoading: false, blurhashUrl: null }); (0, _defineProperty2.default)(this, "videoOnPlay", async () => { if (this.hasContentUrl() || this.state.fetchingData || this.state.error) { // We have the file, we are fetching the file, or there is an error. return; } this.setState({ // To stop subsequent download attempts fetchingData: true }); if (!this.props.mediaEventHelper.media.isEncrypted) { this.setState({ error: "No file given in content" }); return; } this.setState({ decryptedUrl: await this.props.mediaEventHelper.sourceUrl.value, decryptedBlob: await this.props.mediaEventHelper.sourceBlob.value, fetchingData: false }, () => { if (!this.videoRef.current) return; this.videoRef.current.play(); }); this.props.onHeightChanged?.(); }); (0, _defineProperty2.default)(this, "getFileBody", () => { if (this.props.forExport) return null; return this.showFileBody && /*#__PURE__*/_react.default.createElement(_MFileBody.default, (0, _extends2.default)({}, this.props, { showGenericPlaceholder: false })); }); } getContentUrl() { const content = this.props.mxEvent.getContent(); // During export, the content url will point to the MSC, which will later point to a local url if (this.props.forExport) return content.file?.url ?? content.url; const media = (0, _Media.mediaFromContent)(content); if (media.isEncrypted) { return this.state.decryptedUrl ?? undefined; } else { return media.srcHttp ?? undefined; } } hasContentUrl() { const url = this.getContentUrl(); return !!url && !url.startsWith("data:"); } getThumbUrl() { // there's no need of thumbnail when the content is local if (this.props.forExport) return null; const content = this.props.mxEvent.getContent(); const media = (0, _Media.mediaFromContent)(content); if (media.isEncrypted && this.state.decryptedThumbnailUrl) { return this.state.decryptedThumbnailUrl; } else if (this.state.posterLoading) { return this.state.blurhashUrl; } else if (media.hasThumbnail) { return media.thumbnailHttp; } else { return null; } } loadBlurhash() { const info = this.props.mxEvent.getContent()?.info; if (!info[_imageMedia.BLURHASH_FIELD]) return; const canvas = document.createElement("canvas"); const { w: width, h: height } = (0, _ImageSize.suggestedSize)(_SettingsStore.default.getValue("Images.size"), { w: info.w, h: info.h }); canvas.width = width; canvas.height = height; const pixels = (0, _blurhash.decode)(info[_imageMedia.BLURHASH_FIELD], width, height); const ctx = canvas.getContext("2d"); const imgData = ctx.createImageData(width, height); imgData.data.set(pixels); ctx.putImageData(imgData, 0, 0); this.setState({ blurhashUrl: canvas.toDataURL(), posterLoading: true }); const content = this.props.mxEvent.getContent(); const media = (0, _Media.mediaFromContent)(content); if (media.hasThumbnail) { const image = new Image(); image.onload = () => { this.setState({ posterLoading: false }); }; image.src = media.thumbnailHttp; } } async componentDidMount() { this.sizeWatcher = _SettingsStore.default.watchSetting("Images.size", null, () => { this.forceUpdate(); // we don't really have a reliable thing to update, so just update the whole thing }); try { this.loadBlurhash(); } catch (e) { _logger.logger.error("Failed to load blurhash", e); } if (this.props.mediaEventHelper?.media.isEncrypted && this.state.decryptedUrl === null) { try { const autoplay = _SettingsStore.default.getValue("autoplayVideo"); const thumbnailUrl = await this.props.mediaEventHelper.thumbnailUrl.value; if (autoplay) { _logger.logger.log("Preloading video"); this.setState({ decryptedUrl: await this.props.mediaEventHelper.sourceUrl.value, decryptedThumbnailUrl: thumbnailUrl, decryptedBlob: await this.props.mediaEventHelper.sourceBlob.value }); this.props.onHeightChanged?.(); } else { _logger.logger.log("NOT preloading video"); const content = this.props.mxEvent.getContent(); let mimetype = content?.info?.mimetype; // clobber quicktime muxed files to be considered MP4 so browsers // are willing to play them if (mimetype == "video/quicktime") { mimetype = "video/mp4"; } this.setState({ // For Chrome and Electron, we need to set some non-empty `src` to // enable the play button. Firefox does not seem to care either // way, so it's fine to do for all browsers. decryptedUrl: `data:${mimetype},`, decryptedThumbnailUrl: thumbnailUrl || `data:${mimetype},`, decryptedBlob: null }); } } catch (err) { _logger.logger.warn("Unable to decrypt attachment: ", err); // Set a placeholder image when we can't decrypt the image. this.setState({ error: err }); } } } componentWillUnmount() { if (this.sizeWatcher) _SettingsStore.default.unwatchSetting(this.sizeWatcher); } get showFileBody() { return this.context.timelineRenderingType !== _RoomContext.TimelineRenderingType.Room && this.context.timelineRenderingType !== _RoomContext.TimelineRenderingType.Pinned && this.context.timelineRenderingType !== _RoomContext.TimelineRenderingType.Search; } render() { const content = this.props.mxEvent.getContent(); const autoplay = !this.props.inhibitInteraction && _SettingsStore.default.getValue("autoplayVideo"); let aspectRatio; if (content.info?.w && content.info?.h) { aspectRatio = `${content.info.w}/${content.info.h}`; } const { w: maxWidth, h: maxHeight } = (0, _ImageSize.suggestedSize)(_SettingsStore.default.getValue("Images.size"), { w: content.info?.w, h: content.info?.h }); // HACK: This div fills out space while the video loads, to prevent scroll jumps const spaceFiller = /*#__PURE__*/_react.default.createElement("div", { style: { width: maxWidth, height: maxHeight } }); if (this.state.error !== null) { return /*#__PURE__*/_react.default.createElement(_MediaProcessingError.default, { className: "mx_MVideoBody" }, (0, _languageHandler._t)("timeline|m.video|error_decrypting")); } // Important: If we aren't autoplaying and we haven't decrypted it yet, show a video with a poster. if (!this.props.forExport && content.file !== undefined && this.state.decryptedUrl === null && autoplay) { // Need to decrypt the attachment // The attachment is decrypted in componentDidMount. // For now show a spinner. return /*#__PURE__*/_react.default.createElement("span", { className: "mx_MVideoBody" }, /*#__PURE__*/_react.default.createElement("div", { className: "mx_MVideoBody_container", style: { maxWidth, maxHeight, aspectRatio } }, /*#__PURE__*/_react.default.createElement(_InlineSpinner.default, null)), spaceFiller); } const contentUrl = this.getContentUrl(); const thumbUrl = this.getThumbUrl(); let poster; let preload = "metadata"; if (content.info && thumbUrl) { poster = thumbUrl; preload = "none"; } const fileBody = this.getFileBody(); return /*#__PURE__*/_react.default.createElement("span", { className: "mx_MVideoBody" }, /*#__PURE__*/_react.default.createElement("div", { className: "mx_MVideoBody_container", style: { maxWidth, maxHeight, aspectRatio } }, /*#__PURE__*/_react.default.createElement("video", { className: "mx_MVideoBody", ref: this.videoRef, src: contentUrl, title: content.body, controls: !this.props.inhibitInteraction // Disable downloading as it doesn't work with e2ee video, // users should use the dedicated Download button in the Message Action Bar , controlsList: "nodownload", preload: preload, muted: autoplay, autoPlay: autoplay, poster: poster, onPlay: this.videoOnPlay }), spaceFiller), fileBody); } } exports.default = MVideoBody; (0, _defineProperty2.default)(MVideoBody, "contextType", _RoomContext.default); //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireDefault","require","_blurhash","_logger","_languageHandler","_SettingsStore","_InlineSpinner","_Media","_imageMedia","_MFileBody","_ImageSize","_RoomContext","_interopRequireWildcard","_MediaProcessingError","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","MVideoBody","React","PureComponent","constructor","args","_defineProperty2","createRef","fetchingData","decryptedUrl","decryptedThumbnailUrl","decryptedBlob","error","posterLoading","blurhashUrl","hasContentUrl","state","setState","props","mediaEventHelper","media","isEncrypted","sourceUrl","value","sourceBlob","videoRef","current","play","onHeightChanged","forExport","showFileBody","createElement","_extends2","showGenericPlaceholder","getContentUrl","content","mxEvent","getContent","file","url","mediaFromContent","undefined","srcHttp","startsWith","getThumbUrl","hasThumbnail","thumbnailHttp","loadBlurhash","info","BLURHASH_FIELD","canvas","document","w","width","h","height","suggestedVideoSize","SettingsStore","getValue","pixels","decode","ctx","getContext","imgData","createImageData","data","putImageData","toDataURL","image","Image","onload","src","componentDidMount","sizeWatcher","watchSetting","forceUpdate","logger","autoplay","thumbnailUrl","log","mimetype","err","warn","componentWillUnmount","unwatchSetting","context","timelineRenderingType","TimelineRenderingType","Room","Pinned","Search","render","inhibitInteraction","aspectRatio","maxWidth","maxHeight","spaceFiller","style","className","_t","contentUrl","thumbUrl","poster","preload","fileBody","getFileBody","ref","title","body","controls","controlsList","muted","autoPlay","onPlay","videoOnPlay","exports","RoomContext"],"sources":["../../../../src/components/views/messages/MVideoBody.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2015-2021 The Matrix.org Foundation C.I.C.\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n*/\n\nimport React, { ReactNode } from \"react\";\nimport { decode } from \"blurhash\";\nimport { MediaEventContent } from \"matrix-js-sdk/src/types\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\n\nimport { _t } from \"../../../languageHandler\";\nimport SettingsStore from \"../../../settings/SettingsStore\";\nimport InlineSpinner from \"../elements/InlineSpinner\";\nimport { mediaFromContent } from \"../../../customisations/Media\";\nimport { BLURHASH_FIELD } from \"../../../utils/image-media\";\nimport { IBodyProps } from \"./IBodyProps\";\nimport MFileBody from \"./MFileBody\";\nimport { ImageSize, suggestedSize as suggestedVideoSize } from \"../../../settings/enums/ImageSize\";\nimport RoomContext, { TimelineRenderingType } from \"../../../contexts/RoomContext\";\nimport MediaProcessingError from \"./shared/MediaProcessingError\";\n\ninterface IState {\n    decryptedUrl: string | null;\n    decryptedThumbnailUrl: string | null;\n    decryptedBlob: Blob | null;\n    error?: any;\n    fetchingData: boolean;\n    posterLoading: boolean;\n    blurhashUrl: string | null;\n}\n\nexport default class MVideoBody extends React.PureComponent<IBodyProps, IState> {\n    public static contextType = RoomContext;\n    public declare context: React.ContextType<typeof RoomContext>;\n\n    private videoRef = React.createRef<HTMLVideoElement>();\n    private sizeWatcher?: string;\n\n    public state = {\n        fetchingData: false,\n        decryptedUrl: null,\n        decryptedThumbnailUrl: null,\n        decryptedBlob: null,\n        error: null,\n        posterLoading: false,\n        blurhashUrl: null,\n    };\n\n    private getContentUrl(): string | undefined {\n        const content = this.props.mxEvent.getContent<MediaEventContent>();\n        // During export, the content url will point to the MSC, which will later point to a local url\n        if (this.props.forExport) return content.file?.url ?? content.url;\n        const media = mediaFromContent(content);\n        if (media.isEncrypted) {\n            return this.state.decryptedUrl ?? undefined;\n        } else {\n            return media.srcHttp ?? undefined;\n        }\n    }\n\n    private hasContentUrl(): boolean {\n        const url = this.getContentUrl();\n        return !!url && !url.startsWith(\"data:\");\n    }\n\n    private getThumbUrl(): string | null {\n        // there's no need of thumbnail when the content is local\n        if (this.props.forExport) return null;\n\n        const content = this.props.mxEvent.getContent<MediaEventContent>();\n        const media = mediaFromContent(content);\n\n        if (media.isEncrypted && this.state.decryptedThumbnailUrl) {\n            return this.state.decryptedThumbnailUrl;\n        } else if (this.state.posterLoading) {\n            return this.state.blurhashUrl;\n        } else if (media.hasThumbnail) {\n            return media.thumbnailHttp;\n        } else {\n            return null;\n        }\n    }\n\n    private loadBlurhash(): void {\n        const info = this.props.mxEvent.getContent()?.info;\n        if (!info[BLURHASH_FIELD]) return;\n\n        const canvas = document.createElement(\"canvas\");\n\n        const { w: width, h: height } = suggestedVideoSize(SettingsStore.getValue(\"Images.size\") as ImageSize, {\n            w: info.w,\n            h: info.h,\n        });\n\n        canvas.width = width;\n        canvas.height = height;\n\n        const pixels = decode(info[BLURHASH_FIELD], width, height);\n        const ctx = canvas.getContext(\"2d\")!;\n        const imgData = ctx.createImageData(width, height);\n        imgData.data.set(pixels);\n        ctx.putImageData(imgData, 0, 0);\n\n        this.setState({\n            blurhashUrl: canvas.toDataURL(),\n            posterLoading: true,\n        });\n\n        const content = this.props.mxEvent.getContent<MediaEventContent>();\n        const media = mediaFromContent(content);\n        if (media.hasThumbnail) {\n            const image = new Image();\n            image.onload = () => {\n                this.setState({ posterLoading: false });\n            };\n            image.src = media.thumbnailHttp!;\n        }\n    }\n\n    public async componentDidMount(): Promise<void> {\n        this.sizeWatcher = SettingsStore.watchSetting(\"Images.size\", null, () => {\n            this.forceUpdate(); // we don't really have a reliable thing to update, so just update the whole thing\n        });\n\n        try {\n            this.loadBlurhash();\n        } catch (e) {\n            logger.error(\"Failed to load blurhash\", e);\n        }\n\n        if (this.props.mediaEventHelper?.media.isEncrypted && this.state.decryptedUrl === null) {\n            try {\n                const autoplay = SettingsStore.getValue(\"autoplayVideo\") as boolean;\n                const thumbnailUrl = await this.props.mediaEventHelper.thumbnailUrl.value;\n                if (autoplay) {\n                    logger.log(\"Preloading video\");\n                    this.setState({\n                        decryptedUrl: await this.props.mediaEventHelper.sourceUrl.value,\n                        decryptedThumbnailUrl: thumbnailUrl,\n                        decryptedBlob: await this.props.mediaEventHelper.sourceBlob.value,\n                    });\n                    this.props.onHeightChanged?.();\n                } else {\n                    logger.log(\"NOT preloading video\");\n                    const content = this.props.mxEvent.getContent<MediaEventContent>();\n\n                    let mimetype = content?.info?.mimetype;\n\n                    // clobber quicktime muxed files to be considered MP4 so browsers\n                    // are willing to play them\n                    if (mimetype == \"video/quicktime\") {\n                        mimetype = \"video/mp4\";\n                    }\n\n                    this.setState({\n                        // For Chrome and Electron, we need to set some non-empty `src` to\n                        // enable the play button. Firefox does not seem to care either\n                        // way, so it's fine to do for all browsers.\n                        decryptedUrl: `data:${mimetype},`,\n                        decryptedThumbnailUrl: thumbnailUrl || `data:${mimetype},`,\n                        decryptedBlob: null,\n                    });\n                }\n            } catch (err) {\n                logger.warn(\"Unable to decrypt attachment: \", err);\n                // Set a placeholder image when we can't decrypt the image.\n                this.setState({\n                    error: err,\n                });\n            }\n        }\n    }\n\n    public componentWillUnmount(): void {\n        if (this.sizeWatcher) SettingsStore.unwatchSetting(this.sizeWatcher);\n    }\n\n    private videoOnPlay = async (): Promise<void> => {\n        if (this.hasContentUrl() || this.state.fetchingData || this.state.error) {\n            // We have the file, we are fetching the file, or there is an error.\n            return;\n        }\n        this.setState({\n            // To stop subsequent download attempts\n            fetchingData: true,\n        });\n        if (!this.props.mediaEventHelper!.media.isEncrypted) {\n            this.setState({\n                error: \"No file given in content\",\n            });\n            return;\n        }\n        this.setState(\n            {\n                decryptedUrl: await this.props.mediaEventHelper!.sourceUrl.value,\n                decryptedBlob: await this.props.mediaEventHelper!.sourceBlob.value,\n                fetchingData: false,\n            },\n            () => {\n                if (!this.videoRef.current) return;\n                this.videoRef.current.play();\n            },\n        );\n        this.props.onHeightChanged?.();\n    };\n\n    protected get showFileBody(): boolean {\n        return (\n            this.context.timelineRenderingType !== TimelineRenderingType.Room &&\n            this.context.timelineRenderingType !== TimelineRenderingType.Pinned &&\n            this.context.timelineRenderingType !== TimelineRenderingType.Search\n        );\n    }\n\n    private getFileBody = (): ReactNode => {\n        if (this.props.forExport) return null;\n        return this.showFileBody && <MFileBody {...this.props} showGenericPlaceholder={false} />;\n    };\n\n    public render(): React.ReactNode {\n        const content = this.props.mxEvent.getContent();\n        const autoplay = !this.props.inhibitInteraction && SettingsStore.getValue(\"autoplayVideo\");\n\n        let aspectRatio;\n        if (content.info?.w && content.info?.h) {\n            aspectRatio = `${content.info.w}/${content.info.h}`;\n        }\n        const { w: maxWidth, h: maxHeight } = suggestedVideoSize(SettingsStore.getValue(\"Images.size\") as ImageSize, {\n            w: content.info?.w,\n            h: content.info?.h,\n        });\n\n        // HACK: This div fills out space while the video loads, to prevent scroll jumps\n        const spaceFiller = <div style={{ width: maxWidth, height: maxHeight }} />;\n\n        if (this.state.error !== null) {\n            return (\n                <MediaProcessingError className=\"mx_MVideoBody\">\n                    {_t(\"timeline|m.video|error_decrypting\")}\n                </MediaProcessingError>\n            );\n        }\n\n        // Important: If we aren't autoplaying and we haven't decrypted it yet, show a video with a poster.\n        if (!this.props.forExport && content.file !== undefined && this.state.decryptedUrl === null && autoplay) {\n            // Need to decrypt the attachment\n            // The attachment is decrypted in componentDidMount.\n            // For now show a spinner.\n            return (\n                <span className=\"mx_MVideoBody\">\n                    <div className=\"mx_MVideoBody_container\" style={{ maxWidth, maxHeight, aspectRatio }}>\n                        <InlineSpinner />\n                    </div>\n                    {spaceFiller}\n                </span>\n            );\n        }\n\n        const contentUrl = this.getContentUrl();\n        const thumbUrl = this.getThumbUrl();\n        let poster: string | undefined;\n        let preload = \"metadata\";\n        if (content.info && thumbUrl) {\n            poster = thumbUrl;\n            preload = \"none\";\n        }\n\n        const fileBody = this.getFileBody();\n        return (\n            <span className=\"mx_MVideoBody\">\n                <div className=\"mx_MVideoBody_container\" style={{ maxWidth, maxHeight, aspectRatio }}>\n                    <video\n                        className=\"mx_MVideoBody\"\n                        ref={this.videoRef}\n                        src={contentUrl}\n                        title={content.body}\n                        controls={!this.props.inhibitInteraction}\n                        // Disable downloading as it doesn't work with e2ee video,\n                        // users should use the dedicated Download button in the Message Action Bar\n                        controlsList=\"nodownload\"\n                        preload={preload}\n                        muted={autoplay}\n                        autoPlay={autoplay}\n                        poster={poster}\n                        onPlay={this.videoOnPlay}\n                    />\n                    {spaceFiller}\n                </div>\n                {fileBody}\n            </span>\n        );\n    }\n}\n"],"mappings":"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","ignoreList":[]}