UNPKG

matrix-react-sdk

Version:
192 lines (149 loc) 21.7 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _react = _interopRequireWildcard(require("react")); var _propTypes = _interopRequireDefault(require("prop-types")); var _htmlEntities = require("html-entities"); var _HtmlUtils = require("../../../HtmlUtils"); var _SettingsStore = _interopRequireDefault(require("../../../settings/SettingsStore")); var _MatrixClientPeg = require("../../../MatrixClientPeg"); var sdk = _interopRequireWildcard(require("../../../index")); var _Modal = _interopRequireDefault(require("../../../Modal")); var ImageUtils = _interopRequireWildcard(require("../../../ImageUtils")); var _languageHandler = require("../../../languageHandler"); var _replaceableComponent = require("../../../utils/replaceableComponent"); var _Media = require("../../../customisations/Media"); var _dec, _class, _class2, _temp; let LinkPreviewWidget = (_dec = (0, _replaceableComponent.replaceableComponent)("views.rooms.LinkPreviewWidget"), _dec(_class = (_temp = _class2 = class LinkPreviewWidget extends _react.default.Component { constructor(props) { super(props); (0, _defineProperty2.default)(this, "onImageClick", ev => { const p = this.state.preview; if (ev.button != 0 || ev.metaKey) return; ev.preventDefault(); const ImageView = sdk.getComponent("elements.ImageView"); let src = p["og:image"]; if (src && src.startsWith("mxc://")) { src = (0, _Media.mediaFromMxc)(src).srcHttp; } const params = { src: src, width: p["og:image:width"], height: p["og:image:height"], name: p["og:title"] || p["og:description"] || this.props.link, fileSize: p["matrix:image:size"], link: this.props.link }; _Modal.default.createDialog(ImageView, params, "mx_Dialog_lightbox", null, true); }); this.state = { preview: null }; this.unmounted = false; _MatrixClientPeg.MatrixClientPeg.get().getUrlPreview(this.props.link, this.props.mxEvent.getTs()).then(res => { if (this.unmounted) { return; } this.setState({ preview: res }, this.props.onHeightChanged); }, error => { console.error("Failed to get URL preview: " + error); }); this._description = /*#__PURE__*/(0, _react.createRef)(); } componentDidMount() { if (this._description.current) { (0, _HtmlUtils.linkifyElement)(this._description.current); } } componentDidUpdate() { if (this._description.current) { (0, _HtmlUtils.linkifyElement)(this._description.current); } } componentWillUnmount() { this.unmounted = true; } render() { const p = this.state.preview; if (!p || Object.keys(p).length === 0) { return /*#__PURE__*/_react.default.createElement("div", null); } // FIXME: do we want to factor out all image displaying between this and MImageBody - especially for lightboxing? let image = p["og:image"]; if (!_SettingsStore.default.getValue("showImages")) { image = null; // Don't render a button to show the image, just hide it outright } const imageMaxWidth = 100; const imageMaxHeight = 100; if (image && image.startsWith("mxc://")) { // We deliberately don't want a square here, so use the source HTTP thumbnail function image = (0, _Media.mediaFromMxc)(image).getThumbnailOfSourceHttp(imageMaxWidth, imageMaxHeight, 'scale'); } let thumbHeight = imageMaxHeight; if (p["og:image:width"] && p["og:image:height"]) { thumbHeight = ImageUtils.thumbHeight(p["og:image:width"], p["og:image:height"], imageMaxWidth, imageMaxHeight); } let img; if (image) { img = /*#__PURE__*/_react.default.createElement("div", { className: "mx_LinkPreviewWidget_image", style: { height: thumbHeight } }, /*#__PURE__*/_react.default.createElement("img", { style: { maxWidth: imageMaxWidth, maxHeight: imageMaxHeight }, src: image, onClick: this.onImageClick })); } // The description includes &-encoded HTML entities, we decode those as React treats the thing as an // opaque string. This does not allow any HTML to be injected into the DOM. const description = _htmlEntities.AllHtmlEntities.decode(p["og:description"] || ""); const AccessibleButton = sdk.getComponent('elements.AccessibleButton'); return /*#__PURE__*/_react.default.createElement("div", { className: "mx_LinkPreviewWidget" }, img, /*#__PURE__*/_react.default.createElement("div", { className: "mx_LinkPreviewWidget_caption" }, /*#__PURE__*/_react.default.createElement("div", { className: "mx_LinkPreviewWidget_title" }, /*#__PURE__*/_react.default.createElement("a", { href: this.props.link, target: "_blank", rel: "noreferrer noopener" }, p["og:title"])), /*#__PURE__*/_react.default.createElement("div", { className: "mx_LinkPreviewWidget_siteName" }, p["og:site_name"] ? " - " + p["og:site_name"] : null), /*#__PURE__*/_react.default.createElement("div", { className: "mx_LinkPreviewWidget_description", ref: this._description }, description)), /*#__PURE__*/_react.default.createElement(AccessibleButton, { className: "mx_LinkPreviewWidget_cancel", onClick: this.props.onCancelClick, "aria-label": (0, _languageHandler._t)("Close preview") }, /*#__PURE__*/_react.default.createElement("img", { className: "mx_filterFlipColor", alt: "", role: "presentation", src: require("../../../../res/img/cancel.svg"), width: "18", height: "18" }))); } }, (0, _defineProperty2.default)(_class2, "propTypes", { link: _propTypes.default.string.isRequired, // the URL being previewed mxEvent: _propTypes.default.object.isRequired, // the Event associated with the preview onCancelClick: _propTypes.default.func, // called when the preview's cancel ('hide') button is clicked onHeightChanged: _propTypes.default.func // called when the preview's contents has loaded }), _temp)) || _class); exports.default = LinkPreviewWidget; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/rooms/LinkPreviewWidget.js"],"names":["LinkPreviewWidget","React","Component","constructor","props","ev","p","state","preview","button","metaKey","preventDefault","ImageView","sdk","getComponent","src","startsWith","srcHttp","params","width","height","name","link","fileSize","Modal","createDialog","unmounted","MatrixClientPeg","get","getUrlPreview","mxEvent","getTs","then","res","setState","onHeightChanged","error","console","_description","componentDidMount","current","componentDidUpdate","componentWillUnmount","render","Object","keys","length","image","SettingsStore","getValue","imageMaxWidth","imageMaxHeight","getThumbnailOfSourceHttp","thumbHeight","ImageUtils","img","maxWidth","maxHeight","onImageClick","description","AllHtmlEntities","decode","AccessibleButton","onCancelClick","require","PropTypes","string","isRequired","object","func"],"mappings":"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","sourcesContent":["/*\nCopyright 2016 OpenMarket Ltd\nCopyright 2019 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n    http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\nimport React, {createRef} from 'react';\nimport PropTypes from 'prop-types';\nimport { AllHtmlEntities } from 'html-entities';\nimport {linkifyElement} from '../../../HtmlUtils';\nimport SettingsStore from \"../../../settings/SettingsStore\";\nimport {MatrixClientPeg} from \"../../../MatrixClientPeg\";\nimport * as sdk from \"../../../index\";\nimport Modal from \"../../../Modal\";\nimport * as ImageUtils from \"../../../ImageUtils\";\nimport { _t } from \"../../../languageHandler\";\nimport {replaceableComponent} from \"../../../utils/replaceableComponent\";\nimport {mediaFromMxc} from \"../../../customisations/Media\";\n\n@replaceableComponent(\"views.rooms.LinkPreviewWidget\")\nexport default class LinkPreviewWidget extends React.Component {\n    static propTypes = {\n        link: PropTypes.string.isRequired, // the URL being previewed\n        mxEvent: PropTypes.object.isRequired, // the Event associated with the preview\n        onCancelClick: PropTypes.func, // called when the preview's cancel ('hide') button is clicked\n        onHeightChanged: PropTypes.func, // called when the preview's contents has loaded\n    };\n\n    constructor(props) {\n        super(props);\n\n        this.state = {\n            preview: null,\n        };\n\n        this.unmounted = false;\n        MatrixClientPeg.get().getUrlPreview(this.props.link, this.props.mxEvent.getTs()).then((res)=>{\n            if (this.unmounted) {\n                return;\n            }\n            this.setState(\n                { preview: res },\n                this.props.onHeightChanged,\n            );\n        }, (error)=>{\n            console.error(\"Failed to get URL preview: \" + error);\n        });\n\n        this._description = createRef();\n    }\n\n    componentDidMount() {\n        if (this._description.current) {\n            linkifyElement(this._description.current);\n        }\n    }\n\n    componentDidUpdate() {\n        if (this._description.current) {\n            linkifyElement(this._description.current);\n        }\n    }\n\n    componentWillUnmount() {\n        this.unmounted = true;\n    }\n\n    onImageClick = ev => {\n        const p = this.state.preview;\n        if (ev.button != 0 || ev.metaKey) return;\n        ev.preventDefault();\n        const ImageView = sdk.getComponent(\"elements.ImageView\");\n\n        let src = p[\"og:image\"];\n        if (src && src.startsWith(\"mxc://\")) {\n            src = mediaFromMxc(src).srcHttp;\n        }\n\n        const params = {\n            src: src,\n            width: p[\"og:image:width\"],\n            height: p[\"og:image:height\"],\n            name: p[\"og:title\"] || p[\"og:description\"] || this.props.link,\n            fileSize: p[\"matrix:image:size\"],\n            link: this.props.link,\n        };\n\n        Modal.createDialog(ImageView, params, \"mx_Dialog_lightbox\", null, true);\n    };\n\n    render() {\n        const p = this.state.preview;\n        if (!p || Object.keys(p).length === 0) {\n            return <div />;\n        }\n\n        // FIXME: do we want to factor out all image displaying between this and MImageBody - especially for lightboxing?\n        let image = p[\"og:image\"];\n        if (!SettingsStore.getValue(\"showImages\")) {\n            image = null; // Don't render a button to show the image, just hide it outright\n        }\n        const imageMaxWidth = 100;\n        const imageMaxHeight = 100;\n        if (image && image.startsWith(\"mxc://\")) {\n            // We deliberately don't want a square here, so use the source HTTP thumbnail function\n            image = mediaFromMxc(image).getThumbnailOfSourceHttp(imageMaxWidth, imageMaxHeight, 'scale');\n        }\n\n        let thumbHeight = imageMaxHeight;\n        if (p[\"og:image:width\"] && p[\"og:image:height\"]) {\n            thumbHeight = ImageUtils.thumbHeight(\n                p[\"og:image:width\"], p[\"og:image:height\"],\n                imageMaxWidth, imageMaxHeight,\n            );\n        }\n\n        let img;\n        if (image) {\n            img = <div className=\"mx_LinkPreviewWidget_image\" style={{ height: thumbHeight }}>\n                <img style={{ maxWidth: imageMaxWidth, maxHeight: imageMaxHeight }} src={image} onClick={this.onImageClick} />\n            </div>;\n        }\n\n        // The description includes &-encoded HTML entities, we decode those as React treats the thing as an\n        // opaque string. This does not allow any HTML to be injected into the DOM.\n        const description = AllHtmlEntities.decode(p[\"og:description\"] || \"\");\n\n        const AccessibleButton = sdk.getComponent('elements.AccessibleButton');\n        return (\n            <div className=\"mx_LinkPreviewWidget\">\n                { img }\n                <div className=\"mx_LinkPreviewWidget_caption\">\n                    <div className=\"mx_LinkPreviewWidget_title\"><a href={this.props.link} target=\"_blank\" rel=\"noreferrer noopener\">{ p[\"og:title\"] }</a></div>\n                    <div className=\"mx_LinkPreviewWidget_siteName\">{ p[\"og:site_name\"] ? (\" - \" + p[\"og:site_name\"]) : null }</div>\n                    <div className=\"mx_LinkPreviewWidget_description\" ref={this._description}>\n                        { description }\n                    </div>\n                </div>\n                <AccessibleButton className=\"mx_LinkPreviewWidget_cancel\" onClick={this.props.onCancelClick} aria-label={_t(\"Close preview\")}>\n                    <img className=\"mx_filterFlipColor\" alt=\"\" role=\"presentation\"\n                        src={require(\"../../../../res/img/cancel.svg\")} width=\"18\" height=\"18\" />\n                </AccessibleButton>\n            </div>\n        );\n    }\n}\n"]}