UNPKG

matrix-react-sdk

Version:
111 lines (105 loc) 16 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _react = _interopRequireDefault(require("react")); var _reactDom = _interopRequireDefault(require("react-dom")); /* Copyright 2024 New Vector Ltd. Copyright 2022 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. */ function isReactElement(c) { return typeof c === "object" && "type" in c; } /** * The NodeAnimator contains components and animates transitions. * It will only pick up direct changes to properties ('left', currently), and so * will not work for animating positional changes where the position is implicit * from DOM order. This makes it a lot simpler and lighter: if you need fully * automatic positional animation, look at react-shuffle or similar libraries. */ class NodeAnimator extends _react.default.Component { constructor(props) { super(props); (0, _defineProperty2.default)(this, "nodes", {}); (0, _defineProperty2.default)(this, "children", {}); this.updateChildren(this.props.children); } componentDidUpdate() { this.updateChildren(this.props.children); } /** * * @param {HTMLElement} node element to apply styles to * @param {React.CSSProperties} styles a key/value pair of CSS properties * @returns {void} */ applyStyles(node, styles) { Object.entries(styles).forEach(([property, value]) => { node.style[property] = value; }); } updateChildren(newChildren) { const oldChildren = this.children || {}; this.children = {}; _react.default.Children.toArray(newChildren).forEach(c => { if (!isReactElement(c)) return; if (oldChildren[c.key]) { const old = oldChildren[c.key]; const oldNode = _reactDom.default.findDOMNode(this.nodes[old.key]); if (oldNode && oldNode.style.left !== c.props.style.left) { this.applyStyles(oldNode, { left: c.props.style.left }); } // clone the old element with the props (and children) of the new element // so prop updates are still received by the children. this.children[c.key] = /*#__PURE__*/_react.default.cloneElement(old, c.props, c.props.children); } else { // new element. If we have a startStyle, use that as the style and go through // the enter animations const newProps = {}; const restingStyle = c.props.style; const startStyles = this.props.startStyles; if (startStyles.length > 0) { const startStyle = startStyles[0]; newProps.style = startStyle; } newProps.ref = n => this.collectNode(c.key, n, restingStyle); this.children[c.key] = /*#__PURE__*/_react.default.cloneElement(c, newProps); } }); } collectNode(k, node, restingStyle) { if (node && this.nodes[k] === undefined && this.props.startStyles.length > 0) { const startStyles = this.props.startStyles; const domNode = _reactDom.default.findDOMNode(node); // start from startStyle 1: 0 is the one we gave it // to start with, so now we animate 1 etc. for (let i = 1; i < startStyles.length; ++i) { this.applyStyles(domNode, startStyles[i]); } // and then we animate to the resting state window.setTimeout(() => { this.applyStyles(domNode, restingStyle); }, 0); } this.nodes[k] = node; if (this.props.innerRef) { this.props.innerRef.current = node; } } render() { return /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, Object.values(this.children)); } } exports.default = NodeAnimator; (0, _defineProperty2.default)(NodeAnimator, "defaultProps", { startStyles: [] }); //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireDefault","require","_reactDom","isReactElement","c","NodeAnimator","React","Component","constructor","props","_defineProperty2","default","updateChildren","children","componentDidUpdate","applyStyles","node","styles","Object","entries","forEach","property","value","style","newChildren","oldChildren","Children","toArray","key","old","oldNode","ReactDom","findDOMNode","nodes","left","cloneElement","newProps","restingStyle","startStyles","length","startStyle","ref","n","collectNode","k","undefined","domNode","i","window","setTimeout","innerRef","current","render","createElement","Fragment","values","exports"],"sources":["../src/NodeAnimator.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2022 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, { Key, MutableRefObject, ReactElement, ReactFragment, ReactInstance, ReactPortal } from \"react\";\nimport ReactDom from \"react-dom\";\n\ninterface IChildProps {\n    style: React.CSSProperties;\n    ref: (node: React.ReactInstance) => void;\n}\n\ninterface IProps {\n    // either a list of child nodes, or a single child.\n    children: React.ReactNode;\n\n    // optional transition information for changing existing children\n    transition?: object;\n\n    // a list of state objects to apply to each child node in turn\n    startStyles: React.CSSProperties[];\n\n    innerRef?: MutableRefObject<any>;\n}\n\nfunction isReactElement(c: ReactElement | ReactFragment | ReactPortal): c is ReactElement {\n    return typeof c === \"object\" && \"type\" in c;\n}\n\n/**\n * The NodeAnimator contains components and animates transitions.\n * It will only pick up direct changes to properties ('left', currently), and so\n * will not work for animating positional changes where the position is implicit\n * from DOM order. This makes it a lot simpler and lighter: if you need fully\n * automatic positional animation, look at react-shuffle or similar libraries.\n */\nexport default class NodeAnimator extends React.Component<IProps> {\n    private nodes: Record<string, ReactInstance> = {};\n    private children: { [key: string]: ReactElement } = {};\n    public static defaultProps: Partial<IProps> = {\n        startStyles: [],\n    };\n\n    public constructor(props: IProps) {\n        super(props);\n\n        this.updateChildren(this.props.children);\n    }\n\n    public componentDidUpdate(): void {\n        this.updateChildren(this.props.children);\n    }\n\n    /**\n     *\n     * @param {HTMLElement} node element to apply styles to\n     * @param {React.CSSProperties} styles a key/value pair of CSS properties\n     * @returns {void}\n     */\n    private applyStyles(node: HTMLElement, styles: React.CSSProperties): void {\n        Object.entries(styles).forEach(([property, value]) => {\n            node.style[property as keyof Omit<CSSStyleDeclaration, \"length\" | \"parentRule\">] = value;\n        });\n    }\n\n    private updateChildren(newChildren: React.ReactNode): void {\n        const oldChildren = this.children || {};\n        this.children = {};\n        React.Children.toArray(newChildren).forEach((c) => {\n            if (!isReactElement(c)) return;\n            if (oldChildren[c.key!]) {\n                const old = oldChildren[c.key!];\n                const oldNode = ReactDom.findDOMNode(this.nodes[old.key!]);\n\n                if (oldNode && (oldNode as HTMLElement).style.left !== c.props.style.left) {\n                    this.applyStyles(oldNode as HTMLElement, { left: c.props.style.left });\n                }\n                // clone the old element with the props (and children) of the new element\n                // so prop updates are still received by the children.\n                this.children[c.key!] = React.cloneElement(old, c.props, c.props.children);\n            } else {\n                // new element. If we have a startStyle, use that as the style and go through\n                // the enter animations\n                const newProps: Partial<IChildProps> = {};\n                const restingStyle = c.props.style;\n\n                const startStyles = this.props.startStyles;\n                if (startStyles.length > 0) {\n                    const startStyle = startStyles[0];\n                    newProps.style = startStyle;\n                }\n\n                newProps.ref = (n) => this.collectNode(c.key!, n, restingStyle);\n\n                this.children[c.key!] = React.cloneElement(c, newProps);\n            }\n        });\n    }\n\n    private collectNode(k: Key, node: React.ReactInstance, restingStyle: React.CSSProperties): void {\n        if (node && this.nodes[k] === undefined && this.props.startStyles.length > 0) {\n            const startStyles = this.props.startStyles;\n            const domNode = ReactDom.findDOMNode(node);\n            // start from startStyle 1: 0 is the one we gave it\n            // to start with, so now we animate 1 etc.\n            for (let i = 1; i < startStyles.length; ++i) {\n                this.applyStyles(domNode as HTMLElement, startStyles[i]);\n            }\n\n            // and then we animate to the resting state\n            window.setTimeout(() => {\n                this.applyStyles(domNode as HTMLElement, restingStyle);\n            }, 0);\n        }\n        this.nodes[k] = node;\n\n        if (this.props.innerRef) {\n            this.props.innerRef.current = node;\n        }\n    }\n\n    public render(): React.ReactNode {\n        return <>{Object.values(this.children)}</>;\n    }\n}\n"],"mappings":"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","ignoreList":[]}