matrix-react-sdk
Version:
SDK for matrix.org using React
206 lines (163 loc) • 22.1 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.getPersistKey = exports.default = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireDefault(require("react"));
var _reactDom = _interopRequireDefault(require("react-dom"));
var _propTypes = _interopRequireDefault(require("prop-types"));
var _lodash = require("lodash");
var _resizeObserverPolyfill = _interopRequireDefault(require("resize-observer-polyfill"));
var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher"));
var _MatrixClientContext = _interopRequireDefault(require("../../../contexts/MatrixClientContext"));
var _MatrixClientPeg = require("../../../MatrixClientPeg");
var _utils = require("matrix-js-sdk/src/utils");
var _replaceableComponent = require("../../../utils/replaceableComponent");
var _dec, _class, _class2, _temp;
// Shamelessly ripped off Modal.js. There's probably a better way
// of doing reusable widgets like dialog boxes & menus where we go and
// pass in a custom control as the actual body.
function getContainer(containerId) {
return document.getElementById(containerId);
}
function getOrCreateContainer(containerId) {
let container = getContainer(containerId);
if (!container) {
container = document.createElement("div");
container.id = containerId;
document.body.appendChild(container);
}
return container;
}
/*
* Class of component that renders its children in a separate ReactDOM virtual tree
* in a container element appended to document.body.
*
* This prevents the children from being unmounted when the parent of PersistedElement
* unmounts, allowing them to persist.
*
* When PE is unmounted, it hides the children using CSS. When mounted or updated, the
* children are made visible and are positioned into a div that is given the same
* bounding rect as the parent of PE.
*/
let PersistedElement = (_dec = (0, _replaceableComponent.replaceableComponent)("views.elements.PersistedElement"), _dec(_class = (_temp = _class2 = class PersistedElement extends _react.default.Component {
constructor() {
super();
(0, _defineProperty2.default)(this, "updateChildPosition", (0, _lodash.throttle)((child, parent) => {
if (!child || !parent) return;
const parentRect = parent.getBoundingClientRect();
Object.assign(child.style, {
zIndex: (0, _utils.isNullOrUndefined)(this.props.zIndex) ? 9 : this.props.zIndex,
position: 'absolute',
top: parentRect.top + 'px',
left: parentRect.left + 'px',
width: parentRect.width + 'px',
height: parentRect.height + 'px'
});
}, 100, {
trailing: true,
leading: true
}));
this.collectChildContainer = this.collectChildContainer.bind(this);
this.collectChild = this.collectChild.bind(this);
this._repositionChild = this._repositionChild.bind(this);
this._onAction = this._onAction.bind(this);
this.resizeObserver = new _resizeObserverPolyfill.default(this._repositionChild); // Annoyingly, a resize observer is insufficient, since we also care
// about when the element moves on the screen without changing its
// dimensions. Doesn't look like there's a ResizeObserver equivalent
// for this, so we bodge it by listening for document resize and
// the timeline_resize action.
window.addEventListener('resize', this._repositionChild);
this._dispatcherRef = _dispatcher.default.register(this._onAction);
}
/**
* Removes the DOM elements created when a PersistedElement with the given
* persistKey was mounted. The DOM elements will be re-added if another
* PeristedElement is mounted in the future.
*
* @param {string} persistKey Key used to uniquely identify this PersistedElement
*/
static destroyElement(persistKey) {
const container = getContainer('mx_persistedElement_' + persistKey);
if (container) {
container.remove();
}
}
static isMounted(persistKey) {
return Boolean(getContainer('mx_persistedElement_' + persistKey));
}
collectChildContainer(ref) {
if (this.childContainer) {
this.resizeObserver.unobserve(this.childContainer);
}
this.childContainer = ref;
if (ref) {
this.resizeObserver.observe(ref);
}
}
collectChild(ref) {
this.child = ref;
this.updateChild();
}
componentDidMount() {
this.updateChild();
this.renderApp();
}
componentDidUpdate() {
this.updateChild();
this.renderApp();
}
componentWillUnmount() {
this.updateChildVisibility(this.child, false);
this.resizeObserver.disconnect();
window.removeEventListener('resize', this._repositionChild);
_dispatcher.default.unregister(this._dispatcherRef);
}
_onAction(payload) {
if (payload.action === 'timeline_resize') {
this._repositionChild();
} else if (payload.action === 'logout') {
PersistedElement.destroyElement(this.props.persistKey);
}
}
_repositionChild() {
this.updateChildPosition(this.child, this.childContainer);
}
updateChild() {
this.updateChildPosition(this.child, this.childContainer);
this.updateChildVisibility(this.child, true);
}
renderApp() {
const content = /*#__PURE__*/_react.default.createElement(_MatrixClientContext.default.Provider, {
value: _MatrixClientPeg.MatrixClientPeg.get()
}, /*#__PURE__*/_react.default.createElement("div", {
ref: this.collectChild,
style: this.props.style
}, this.props.children));
_reactDom.default.render(content, getOrCreateContainer('mx_persistedElement_' + this.props.persistKey));
}
updateChildVisibility(child, visible) {
if (!child) return;
child.style.display = visible ? 'block' : 'none';
}
render() {
return /*#__PURE__*/_react.default.createElement("div", {
ref: this.collectChildContainer
});
}
}, (0, _defineProperty2.default)(_class2, "propTypes", {
// Unique identifier for this PersistedElement instance
// Any PersistedElements with the same persistKey will use
// the same DOM container.
persistKey: _propTypes.default.string.isRequired,
// z-index for the element. Defaults to 9.
zIndex: _propTypes.default.number
}), _temp)) || _class);
exports.default = PersistedElement;
const getPersistKey = (appId
/*: string*/
) => 'widget_' + appId;
exports.getPersistKey = getPersistKey;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/elements/PersistedElement.js"],"names":["getContainer","containerId","document","getElementById","getOrCreateContainer","container","createElement","id","body","appendChild","PersistedElement","React","Component","constructor","child","parent","parentRect","getBoundingClientRect","Object","assign","style","zIndex","props","position","top","left","width","height","trailing","leading","collectChildContainer","bind","collectChild","_repositionChild","_onAction","resizeObserver","ResizeObserver","window","addEventListener","_dispatcherRef","dis","register","destroyElement","persistKey","remove","isMounted","Boolean","ref","childContainer","unobserve","observe","updateChild","componentDidMount","renderApp","componentDidUpdate","componentWillUnmount","updateChildVisibility","disconnect","removeEventListener","unregister","payload","action","updateChildPosition","content","MatrixClientPeg","get","children","ReactDOM","render","visible","display","PropTypes","string","isRequired","number","getPersistKey","appId"],"mappings":"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","sourcesContent":["/*\nCopyright 2018 New Vector Ltd.\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 from 'react';\nimport ReactDOM from 'react-dom';\nimport PropTypes from 'prop-types';\nimport {throttle} from \"lodash\";\nimport ResizeObserver from 'resize-observer-polyfill';\n\nimport dis from '../../../dispatcher/dispatcher';\nimport MatrixClientContext from \"../../../contexts/MatrixClientContext\";\nimport {MatrixClientPeg} from \"../../../MatrixClientPeg\";\nimport {isNullOrUndefined} from \"matrix-js-sdk/src/utils\";\nimport {replaceableComponent} from \"../../../utils/replaceableComponent\";\n\n// Shamelessly ripped off Modal.js.  There's probably a better way\n// of doing reusable widgets like dialog boxes & menus where we go and\n// pass in a custom control as the actual body.\n\nfunction getContainer(containerId) {\n    return document.getElementById(containerId);\n}\n\nfunction getOrCreateContainer(containerId) {\n    let container = getContainer(containerId);\n\n    if (!container) {\n        container = document.createElement(\"div\");\n        container.id = containerId;\n        document.body.appendChild(container);\n    }\n\n    return container;\n}\n\n/*\n * Class of component that renders its children in a separate ReactDOM virtual tree\n * in a container element appended to document.body.\n *\n * This prevents the children from being unmounted when the parent of PersistedElement\n * unmounts, allowing them to persist.\n *\n * When PE is unmounted, it hides the children using CSS. When mounted or updated, the\n * children are made visible and are positioned into a div that is given the same\n * bounding rect as the parent of PE.\n */\n@replaceableComponent(\"views.elements.PersistedElement\")\nexport default class PersistedElement extends React.Component {\n    static propTypes = {\n        // Unique identifier for this PersistedElement instance\n        // Any PersistedElements with the same persistKey will use\n        // the same DOM container.\n        persistKey: PropTypes.string.isRequired,\n\n        // z-index for the element. Defaults to 9.\n        zIndex: PropTypes.number,\n    };\n\n    constructor() {\n        super();\n        this.collectChildContainer = this.collectChildContainer.bind(this);\n        this.collectChild = this.collectChild.bind(this);\n        this._repositionChild = this._repositionChild.bind(this);\n        this._onAction = this._onAction.bind(this);\n\n        this.resizeObserver = new ResizeObserver(this._repositionChild);\n        // Annoyingly, a resize observer is insufficient, since we also care\n        // about when the element moves on the screen without changing its\n        // dimensions. Doesn't look like there's a ResizeObserver equivalent\n        // for this, so we bodge it by listening for document resize and\n        // the timeline_resize action.\n        window.addEventListener('resize', this._repositionChild);\n        this._dispatcherRef = dis.register(this._onAction);\n    }\n\n    /**\n     * Removes the DOM elements created when a PersistedElement with the given\n     * persistKey was mounted. The DOM elements will be re-added if another\n     * PeristedElement is mounted in the future.\n     *\n     * @param {string} persistKey Key used to uniquely identify this PersistedElement\n     */\n    static destroyElement(persistKey) {\n        const container = getContainer('mx_persistedElement_' + persistKey);\n        if (container) {\n            container.remove();\n        }\n    }\n\n    static isMounted(persistKey) {\n        return Boolean(getContainer('mx_persistedElement_' + persistKey));\n    }\n\n    collectChildContainer(ref) {\n        if (this.childContainer) {\n            this.resizeObserver.unobserve(this.childContainer);\n        }\n        this.childContainer = ref;\n        if (ref) {\n            this.resizeObserver.observe(ref);\n        }\n    }\n\n    collectChild(ref) {\n        this.child = ref;\n        this.updateChild();\n    }\n\n    componentDidMount() {\n        this.updateChild();\n        this.renderApp();\n    }\n\n    componentDidUpdate() {\n        this.updateChild();\n        this.renderApp();\n    }\n\n    componentWillUnmount() {\n        this.updateChildVisibility(this.child, false);\n        this.resizeObserver.disconnect();\n        window.removeEventListener('resize', this._repositionChild);\n        dis.unregister(this._dispatcherRef);\n    }\n\n    _onAction(payload) {\n        if (payload.action === 'timeline_resize') {\n            this._repositionChild();\n        } else if (payload.action === 'logout') {\n            PersistedElement.destroyElement(this.props.persistKey);\n        }\n    }\n\n    _repositionChild() {\n        this.updateChildPosition(this.child, this.childContainer);\n    }\n\n    updateChild() {\n        this.updateChildPosition(this.child, this.childContainer);\n        this.updateChildVisibility(this.child, true);\n    }\n\n    renderApp() {\n        const content = <MatrixClientContext.Provider value={MatrixClientPeg.get()}>\n            <div ref={this.collectChild} style={this.props.style}>\n                {this.props.children}\n            </div>\n        </MatrixClientContext.Provider>;\n\n        ReactDOM.render(content, getOrCreateContainer('mx_persistedElement_'+this.props.persistKey));\n    }\n\n    updateChildVisibility(child, visible) {\n        if (!child) return;\n        child.style.display = visible ? 'block' : 'none';\n    }\n\n    updateChildPosition = throttle((child, parent) => {\n        if (!child || !parent) return;\n\n        const parentRect = parent.getBoundingClientRect();\n        Object.assign(child.style, {\n            zIndex: isNullOrUndefined(this.props.zIndex) ? 9 : this.props.zIndex,\n            position: 'absolute',\n            top: parentRect.top + 'px',\n            left: parentRect.left + 'px',\n            width: parentRect.width + 'px',\n            height: parentRect.height + 'px',\n        });\n    }, 100, {trailing: true, leading: true});\n\n    render() {\n        return <div ref={this.collectChildContainer} />;\n    }\n}\n\nexport const getPersistKey = (appId: string) => 'widget_' + appId;\n"]}