使用 React JS 无限滚动

IT技术 javascript html infinite-scroll reactjs
2021-03-17 16:21:35

我正在寻找使用 React 实现无限滚动的方法。我遇到过react-infinite-scroll并发现它效率低下,因为它只是将节点添加到 DOM 而不会删除它们。是否有任何经过验证的 React 解决方案可以在 DOM 中添加、删除和维护恒定数量的节点。

这是jsfiddle问题。在这个问题中,我希望 DOM 中一次只有 50 个元素。当用户上下滚动时,应该加载和删除其他人。我们开始使用 React 是因为它的优化算法。现在我找不到解决这个问题的方法。我遇到了airbnb 无限 js但它是用 Jquery 实现的。要使用这个 airbnb 无限滚动,我必须放弃我不想做的 React 优化。

我要添加滚动的示例代码是(这里我正在加载所有项目。我的目标是一次仅加载 50 个项目)

/** @jsx React.DOM */

var Hello = React.createClass({
    render: function() {
        return (<li>Hello {this.props.name}</li>);
    }
});

var HelloList = React.createClass({ 
     getInitialState: function() {                            
         var numbers =  [];
         for(var i=1;i<10000;i++){
             numbers.push(i);
         }
         return {data:numbers};
     },

    render: function(){
       var response =  this.state.data.map(function(contact){          
          return (<Hello name="World"></Hello>);
        });

        return (<ul>{response}</ul>)
    }
});

React.renderComponent(<HelloList/>, document.getElementById('content'));

寻求帮助...

3个回答

基本上在滚动时,您要决定哪些元素是可见的,然后重新渲染以仅显示这些元素,顶部和底部有一个间隔元素来表示屏幕外元素。

Vjeux在这里做了一个小提琴,你可以看看:jsfiddle

滚动时执行

scrollState: function(scroll) {
    var visibleStart = Math.floor(scroll / this.state.recordHeight);
    var visibleEnd = Math.min(visibleStart + this.state.recordsPerBody, this.state.total - 1);

    var displayStart = Math.max(0, Math.floor(scroll / this.state.recordHeight) - this.state.recordsPerBody * 1.5);
    var displayEnd = Math.min(displayStart + 4 * this.state.recordsPerBody, this.state.total - 1);

    this.setState({
        visibleStart: visibleStart,
        visibleEnd: visibleEnd,
        displayStart: displayStart,
        displayEnd: displayEnd,
        scroll: scroll
    });
},

然后渲染函数将只显示范围内的行displayStart..displayEnd

您可能还对ReactJS:建模双向无限滚动感兴趣

这是一个伟大的技术......谢谢!但是,当每行的 recordHeight 不同时,它会失败。我正在尝试解决这种情况。如果我让它工作,我会发布它。
2021-04-28 16:21:35
另一个需要检查的项目是 infinity.js(以获得灵感)。如果您有动态高度元素,那么您可以创建“页面”的概念,它是视口中的一组元素。假设有 3 个元素,并且第三个元素超长且超出页面。然后你可以说,“页面高度”是最大的 3 个元素的大小。然后使用最小的元素高度构造虚拟节点所以var count = pageHeight / minElementHeight因此,您可能会构建 50 个元素,即使只渲染了 3 个,但这仍然会给您带来良好的性能。
2021-05-07 16:21:35
@manalang您是否找到了每行不同高度的解决方案?
2021-05-12 16:21:35
@sophie-alpert:是否可以更新 jsfiddle ?我知道你会很忙,但如果你能更新它,它会让像我这样的人受益 :D
2021-05-14 16:21:35
小提琴中什么都没有出现。生成按钮出现,但没有别的。
2021-05-18 16:21:35

查看我们的 React 无限库:

https://github.com/seatgeek/react-infinite

2016 年 12 月更新

实际上,我最近在我的很多项目中都使用了react-virtualized,发现它更好地涵盖了大多数用例。两个库都很好,这取决于您正在寻找什么。例如,react-virtualized 支持通过名为 的 HOC 进行可变高度 JIT 测量CellMeasurer,例如https://bvaughn.github.io/react-virtualized/#/components/CellMeasurer

2018 年 11 月更新

很多来自 react-virtualized 的经验已经移植到来自同一作者的更小、更快、更高效的react-window库中。

@Druska,从技术上讲是的,但是您也可以使用 useWindowAsScrollContainer 选项将窗口用作滚动容器。
2021-04-29 16:21:35
该库仅在您在渲染之前知道元素的高度时才有效。
2021-05-13 16:21:35
react-infinite 库是否支持网格?
2021-05-14 16:21:35
@jos:使用这个库。当 DOM 节点出现在视口中时,它将删除/附加它们。
2021-05-16 16:21:35
import React, { Component } from 'react';
import InfiniteScroll from 'react-infinite-scroller';


const api = {
    baseUrl: '/joblist'
};

class Jobs extends Component {
    constructor(props) {
            super(props);
            this.state = {
                listData: [],
                hasMoreItems: true,
                nextHref: null
        };
    }

    fetchData(){
            var self = this;           
            var url = api.baseUrl;
            if(this.state.nextHref) {
                url = this.state.nextHref;
            }

            fetch(url)
            .then( (response) => {
                return response.json() })   
                    .then( (json) => {
                        var list = self.state.listData;                        
                        json.data.map(data => {
                            list.push(data);
                        });

                        if(json.next_page_url != null) {
                            self.setState({
                                nextHref: resp.next_page_url,
                                listData: list                               
                            });
                        } else {
                            self.setState({
                                hasMoreItems: false
                            });
                        }
                    })
                    .catch(error => console.log('err ' + error));

        }
    }

    componentDidMount() {
       this.fetchData();
    }

    render() {
    const loader = <div className="loader">Loading ...</div>;
    let JobItems; 
    if(this.state.listData){  
        JobItems = this.state.listData.map(Job => {
        return (
            <tr>
                <td>{Job.job_number}</td>
                <td>{Job.title}</td>
                <td>{Job.description}</td>
                <td>{Job.status}</td>
            </tr>
        );
      });
    }
    return (
      <div className="Jobs">
        <div className="container">
            <h2>Jobs List</h2>

            <InfiniteScroll
                pageStart={0}
                loadMore={this.fetchData.bind(this)}
                hasMore={this.state.hasMoreItems}
                loader={loader}>
                <table className="table table-bordered">
                <thead>
                    <tr>
                        <th>Job Number</th>
                        <th>Title</th>
                        <th>Description</th>
                        <th>Status</th>
                    </tr>
                </thead>
                <tbody>
                {JobItems}
                </tbody>
                </table>
            </InfiniteScroll>
        </div>
    </div>
    );
  }

}

export default Jobs;