matrix-react-sdk
Version:
SDK for matrix.org using React
172 lines (166 loc) • 24.8 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 _utils = require("matrix-js-sdk/src/utils");
var _compoundWeb = require("@vector-im/compound-web");
var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher"));
var _MatrixClientContext = _interopRequireDefault(require("../../../contexts/MatrixClientContext"));
var _MatrixClientPeg = require("../../../MatrixClientPeg");
/*
Copyright 2018-2024 New Vector Ltd.
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
const getPersistKey = appId => "widget_" + appId;
// 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.
// We contain all persisted elements within a master container to allow them all to be within the same
// CSS stacking context, and thus be able to control their z-indexes relative to each other.
exports.getPersistKey = getPersistKey;
function getOrCreateMasterContainer() {
let container = getContainer("mx_PersistedElement_container");
if (!container) {
container = document.createElement("div");
container.id = "mx_PersistedElement_container";
document.body.appendChild(container);
}
return container;
}
function getContainer(containerId) {
return document.getElementById(containerId);
}
function getOrCreateContainer(containerId) {
let container = getContainer(containerId);
if (!container) {
container = document.createElement("div");
container.id = containerId;
getOrCreateMasterContainer().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.
*/
class PersistedElement extends _react.default.Component {
constructor(props) {
super(props);
(0, _defineProperty2.default)(this, "resizeObserver", void 0);
(0, _defineProperty2.default)(this, "dispatcherRef", void 0);
(0, _defineProperty2.default)(this, "childContainer", void 0);
(0, _defineProperty2.default)(this, "child", void 0);
(0, _defineProperty2.default)(this, "collectChildContainer", ref => {
if (this.childContainer) {
this.resizeObserver.unobserve(this.childContainer);
}
this.childContainer = ref;
if (ref) {
this.resizeObserver.observe(ref);
}
});
(0, _defineProperty2.default)(this, "collectChild", ref => {
this.child = ref;
this.updateChild();
});
(0, _defineProperty2.default)(this, "onAction", payload => {
if (payload.action === "timeline_resize") {
this.repositionChild();
} else if (payload.action === "logout") {
PersistedElement.destroyElement(this.props.persistKey);
}
});
(0, _defineProperty2.default)(this, "repositionChild", () => {
this.updateChildPosition(this.child, this.childContainer);
});
this.resizeObserver = new ResizeObserver(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);
if (this.props.moveRef) this.props.moveRef.current = this.repositionChild;
}
/**
* Removes the DOM elements created when a PersistedElement with the given
* persistKey was mounted. The DOM elements will be re-added if another
* PersistedElement 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));
}
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);
}
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.safeGet()
}, /*#__PURE__*/_react.default.createElement(_compoundWeb.TooltipProvider, null, /*#__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 = false) {
if (!child) return;
child.style.display = visible ? "block" : "none";
}
updateChildPosition(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: "0",
left: "0",
transform: `translateX(${parentRect.left}px) translateY(${parentRect.top}px)`,
width: parentRect.width + "px",
height: parentRect.height + "px"
});
}
render() {
return /*#__PURE__*/_react.default.createElement("div", {
ref: this.collectChildContainer
});
}
}
exports.default = PersistedElement;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireDefault","require","_reactDom","_utils","_compoundWeb","_dispatcher","_MatrixClientContext","_MatrixClientPeg","getPersistKey","appId","exports","getOrCreateMasterContainer","container","getContainer","document","createElement","id","body","appendChild","containerId","getElementById","getOrCreateContainer","PersistedElement","React","Component","constructor","props","_defineProperty2","default","ref","childContainer","resizeObserver","unobserve","observe","child","updateChild","payload","action","repositionChild","destroyElement","persistKey","updateChildPosition","ResizeObserver","window","addEventListener","dispatcherRef","dis","register","onAction","moveRef","current","remove","isMounted","Boolean","componentDidMount","renderApp","componentDidUpdate","componentWillUnmount","updateChildVisibility","disconnect","removeEventListener","unregister","content","Provider","value","MatrixClientPeg","safeGet","TooltipProvider","collectChild","style","children","ReactDOM","render","visible","display","parent","parentRect","getBoundingClientRect","Object","assign","zIndex","isNullOrUndefined","position","top","left","transform","width","height","collectChildContainer"],"sources":["../../../../src/components/views/elements/PersistedElement.tsx"],"sourcesContent":["/*\nCopyright 2018-2024 New Vector Ltd.\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, { MutableRefObject, ReactNode } from \"react\";\nimport ReactDOM from \"react-dom\";\nimport { isNullOrUndefined } from \"matrix-js-sdk/src/utils\";\nimport { TooltipProvider } from \"@vector-im/compound-web\";\n\nimport dis from \"../../../dispatcher/dispatcher\";\nimport MatrixClientContext from \"../../../contexts/MatrixClientContext\";\nimport { MatrixClientPeg } from \"../../../MatrixClientPeg\";\nimport { ActionPayload } from \"../../../dispatcher/payloads\";\n\nexport const getPersistKey = (appId: string): string => \"widget_\" + appId;\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\n// We contain all persisted elements within a master container to allow them all to be within the same\n// CSS stacking context, and thus be able to control their z-indexes relative to each other.\nfunction getOrCreateMasterContainer(): HTMLDivElement {\n    let container = getContainer(\"mx_PersistedElement_container\");\n    if (!container) {\n        container = document.createElement(\"div\");\n        container.id = \"mx_PersistedElement_container\";\n        document.body.appendChild(container);\n    }\n\n    return container;\n}\n\nfunction getContainer(containerId: string): HTMLDivElement {\n    return document.getElementById(containerId) as HTMLDivElement;\n}\n\nfunction getOrCreateContainer(containerId: string): HTMLDivElement {\n    let container = getContainer(containerId);\n\n    if (!container) {\n        container = document.createElement(\"div\");\n        container.id = containerId;\n        getOrCreateMasterContainer().appendChild(container);\n    }\n\n    return container;\n}\n\ninterface IProps {\n    // Unique identifier for this PersistedElement instance\n    // Any PersistedElements with the same persistKey will use\n    // the same DOM container.\n    persistKey: string;\n\n    // z-index for the element. Defaults to 9.\n    zIndex?: number;\n\n    style?: React.StyleHTMLAttributes<HTMLDivElement>;\n\n    // Handle to manually notify this PersistedElement that it needs to move\n    moveRef?: MutableRefObject<(() => void) | undefined>;\n    children: ReactNode;\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 */\nexport default class PersistedElement extends React.Component<IProps> {\n    private resizeObserver: ResizeObserver;\n    private dispatcherRef: string;\n    private childContainer?: HTMLDivElement;\n    private child?: HTMLDivElement;\n\n    public constructor(props: IProps) {\n        super(props);\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        if (this.props.moveRef) this.props.moveRef.current = this.repositionChild;\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     * PersistedElement is mounted in the future.\n     *\n     * @param {string} persistKey Key used to uniquely identify this PersistedElement\n     */\n    public static destroyElement(persistKey: string): void {\n        const container = getContainer(\"mx_persistedElement_\" + persistKey);\n        if (container) {\n            container.remove();\n        }\n    }\n\n    public static isMounted(persistKey: string): boolean {\n        return Boolean(getContainer(\"mx_persistedElement_\" + persistKey));\n    }\n\n    private collectChildContainer = (ref: HTMLDivElement): void => {\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    private collectChild = (ref: HTMLDivElement): void => {\n        this.child = ref;\n        this.updateChild();\n    };\n\n    public componentDidMount(): void {\n        this.updateChild();\n        this.renderApp();\n    }\n\n    public componentDidUpdate(): void {\n        this.updateChild();\n        this.renderApp();\n    }\n\n    public componentWillUnmount(): void {\n        this.updateChildVisibility(this.child, false);\n        this.resizeObserver.disconnect();\n        window.removeEventListener(\"resize\", this.repositionChild);\n        dis.unregister(this.dispatcherRef);\n    }\n\n    private onAction = (payload: ActionPayload): void => {\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    private repositionChild = (): void => {\n        this.updateChildPosition(this.child, this.childContainer);\n    };\n\n    private updateChild(): void {\n        this.updateChildPosition(this.child, this.childContainer);\n        this.updateChildVisibility(this.child, true);\n    }\n\n    private renderApp(): void {\n        const content = (\n            <MatrixClientContext.Provider value={MatrixClientPeg.safeGet()}>\n                <TooltipProvider>\n                    <div ref={this.collectChild} style={this.props.style}>\n                        {this.props.children}\n                    </div>\n                </TooltipProvider>\n            </MatrixClientContext.Provider>\n        );\n\n        ReactDOM.render(content, getOrCreateContainer(\"mx_persistedElement_\" + this.props.persistKey));\n    }\n\n    private updateChildVisibility(child?: HTMLDivElement, visible = false): void {\n        if (!child) return;\n        child.style.display = visible ? \"block\" : \"none\";\n    }\n\n    private updateChildPosition(child?: HTMLDivElement, parent?: HTMLDivElement): void {\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: \"0\",\n            left: \"0\",\n            transform: `translateX(${parentRect.left}px) translateY(${parentRect.top}px)`,\n            width: parentRect.width + \"px\",\n            height: parentRect.height + \"px\",\n        });\n    }\n\n    public render(): React.ReactNode {\n        return <div ref={this.collectChildContainer} />;\n    }\n}\n"],"mappings":"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","ignoreList":[]}