UNPKG

matrix-react-sdk

Version:
113 lines (110 loc) 16.2 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")); /* Copyright 2019-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. */ class ItemRange { constructor(topCount, renderCount, bottomCount) { this.topCount = topCount; this.renderCount = renderCount; this.bottomCount = bottomCount; } contains(range) { // don't contain empty ranges // as it will prevent clearing the list // once it is scrolled far enough out of view if (!range.renderCount && this.renderCount) { return false; } return range.topCount >= this.topCount && range.topCount + range.renderCount <= this.topCount + this.renderCount; } expand(amount) { // don't expand ranges that won't render anything if (this.renderCount === 0) { return this; } const topGrow = Math.min(amount, this.topCount); const bottomGrow = Math.min(amount, this.bottomCount); return new ItemRange(this.topCount - topGrow, this.renderCount + topGrow + bottomGrow, this.bottomCount - bottomGrow); } totalSize() { return this.topCount + this.renderCount + this.bottomCount; } } class LazyRenderList extends _react.default.Component { constructor(props) { super(props); this.state = LazyRenderList.getDerivedStateFromProps(props, {}); } static getDerivedStateFromProps(props, state) { const range = LazyRenderList.getVisibleRangeFromProps(props); const intersectRange = range.expand(props.overflowMargin); const renderRange = range.expand(props.overflowItems); const listHasChangedSize = !!state.renderRange && renderRange.totalSize() !== state.renderRange.totalSize(); // only update render Range if the list has shrunk/grown and we need to adjust padding OR // if the new range + overflowMargin isn't contained by the old anymore if (listHasChangedSize || !state.renderRange || !state.renderRange.contains(intersectRange)) { return { renderRange }; } return null; } static getVisibleRangeFromProps(props) { const { items, itemHeight, scrollTop, height } = props; const length = items ? items.length : 0; const topCount = Math.min(Math.max(0, Math.floor(scrollTop / itemHeight)), length); const itemsAfterTop = length - topCount; const visibleItems = height !== 0 ? Math.ceil(height / itemHeight) : 0; const renderCount = Math.min(visibleItems, itemsAfterTop); const bottomCount = itemsAfterTop - renderCount; return new ItemRange(topCount, renderCount, bottomCount); } render() { const { itemHeight, items, renderItem } = this.props; const { renderRange } = this.state; const { topCount, renderCount, bottomCount } = renderRange; const paddingTop = topCount * itemHeight; const paddingBottom = bottomCount * itemHeight; const renderedItems = (items || []).slice(topCount, topCount + renderCount); const element = this.props.element || "div"; const elementProps = { style: { paddingTop: `${paddingTop}px`, paddingBottom: `${paddingBottom}px` }, className: this.props.className, role: this.props.role }; return /*#__PURE__*/_react.default.createElement(element, elementProps, renderedItems.map(renderItem)); } } exports.default = LazyRenderList; (0, _defineProperty2.default)(LazyRenderList, "defaultProps", { overflowItems: 20, overflowMargin: 5 }); //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireDefault","require","ItemRange","constructor","topCount","renderCount","bottomCount","contains","range","expand","amount","topGrow","Math","min","bottomGrow","totalSize","LazyRenderList","React","Component","props","state","getDerivedStateFromProps","getVisibleRangeFromProps","intersectRange","overflowMargin","renderRange","overflowItems","listHasChangedSize","items","itemHeight","scrollTop","height","length","max","floor","itemsAfterTop","visibleItems","ceil","render","renderItem","paddingTop","paddingBottom","renderedItems","slice","element","elementProps","style","className","role","createElement","map","exports","default","_defineProperty2"],"sources":["../../../../src/components/views/elements/LazyRenderList.tsx"],"sourcesContent":["/*\nCopyright 2019-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 from \"react\";\n\nclass ItemRange {\n    public constructor(\n        public topCount: number,\n        public renderCount: number,\n        public bottomCount: number,\n    ) {}\n\n    public contains(range: ItemRange): boolean {\n        // don't contain empty ranges\n        // as it will prevent clearing the list\n        // once it is scrolled far enough out of view\n        if (!range.renderCount && this.renderCount) {\n            return false;\n        }\n        return (\n            range.topCount >= this.topCount && range.topCount + range.renderCount <= this.topCount + this.renderCount\n        );\n    }\n\n    public expand(amount: number): ItemRange {\n        // don't expand ranges that won't render anything\n        if (this.renderCount === 0) {\n            return this;\n        }\n\n        const topGrow = Math.min(amount, this.topCount);\n        const bottomGrow = Math.min(amount, this.bottomCount);\n        return new ItemRange(\n            this.topCount - topGrow,\n            this.renderCount + topGrow + bottomGrow,\n            this.bottomCount - bottomGrow,\n        );\n    }\n\n    public totalSize(): number {\n        return this.topCount + this.renderCount + this.bottomCount;\n    }\n}\n\ninterface IProps<T> {\n    // height in pixels of the component returned by `renderItem`\n    itemHeight: number;\n    // function to turn an element of `items` into a react component\n    renderItem: (item: T) => JSX.Element;\n    // scrollTop of the viewport (minus the height of any content above this list like other `LazyRenderList`s)\n    scrollTop: number;\n    // the height of the viewport this content is scrolled in\n    height: number;\n    // all items for the list. These should not be react components, see `renderItem`.\n    items?: T[];\n    // the amount of items to scroll before causing a rerender,\n    // should typically be less than `overflowItems` unless applying\n    // margins in the parent component when using multiple LazyRenderList in one viewport.\n    // use 0 to only rerender when items will come into view.\n    overflowMargin: number;\n    // the amount of items to add at the top and bottom to render,\n    // so not every scroll of causes a rerender.\n    overflowItems: number;\n\n    element?: string;\n    className?: string;\n    role?: string;\n}\n\ninterface IState {\n    renderRange: ItemRange;\n}\n\nexport default class LazyRenderList<T = any> extends React.Component<IProps<T>, IState> {\n    public static defaultProps: Partial<IProps<unknown>> = {\n        overflowItems: 20,\n        overflowMargin: 5,\n    };\n\n    public constructor(props: IProps<T>) {\n        super(props);\n\n        this.state = LazyRenderList.getDerivedStateFromProps(props, {} as IState) as IState;\n    }\n\n    public static getDerivedStateFromProps<T>(props: IProps<T>, state: IState): Partial<IState> | null {\n        const range = LazyRenderList.getVisibleRangeFromProps(props);\n        const intersectRange = range.expand(props.overflowMargin);\n        const renderRange = range.expand(props.overflowItems);\n        const listHasChangedSize = !!state.renderRange && renderRange.totalSize() !== state.renderRange.totalSize();\n        // only update render Range if the list has shrunk/grown and we need to adjust padding OR\n        // if the new range + overflowMargin isn't contained by the old anymore\n        if (listHasChangedSize || !state.renderRange || !state.renderRange.contains(intersectRange)) {\n            return { renderRange };\n        }\n        return null;\n    }\n\n    private static getVisibleRangeFromProps<T>(props: IProps<T>): ItemRange {\n        const { items, itemHeight, scrollTop, height } = props;\n        const length = items ? items.length : 0;\n        const topCount = Math.min(Math.max(0, Math.floor(scrollTop / itemHeight)), length);\n        const itemsAfterTop = length - topCount;\n        const visibleItems = height !== 0 ? Math.ceil(height / itemHeight) : 0;\n        const renderCount = Math.min(visibleItems, itemsAfterTop);\n        const bottomCount = itemsAfterTop - renderCount;\n        return new ItemRange(topCount, renderCount, bottomCount);\n    }\n\n    public render(): React.ReactNode {\n        const { itemHeight, items, renderItem } = this.props;\n        const { renderRange } = this.state;\n        const { topCount, renderCount, bottomCount } = renderRange;\n\n        const paddingTop = topCount * itemHeight;\n        const paddingBottom = bottomCount * itemHeight;\n        const renderedItems = (items || []).slice(topCount, topCount + renderCount);\n\n        const element = this.props.element || \"div\";\n        const elementProps = {\n            style: { paddingTop: `${paddingTop}px`, paddingBottom: `${paddingBottom}px` },\n            className: this.props.className,\n            role: this.props.role,\n        };\n        return React.createElement(element, elementProps, renderedItems.map(renderItem));\n    }\n}\n"],"mappings":"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","ignoreList":[]}