反应固定数据表:未捕获类型错误:this._dataList.getSize 不是函数

Posted

技术标签:

【中文标题】反应固定数据表:未捕获类型错误:this._dataList.getSize 不是函数【英文标题】:React fixed-data-table: Uncaught TypeError: this._dataList.getSize is not a function 【发布时间】:2017-07-12 14:41:58 【问题描述】:

我正在尝试使用 React 开发人员的 this example 来为表格创建搜索过滤器。

我的表格可以静态处理来自后端的数据。我为“样本”数据取出了一个数组,以使搜索功能正常工作。但是我很难理解他们如何使用“假数据”将他们的表填充为seen here,而不是像我想要的那样“只是”用测试数组填充它。

这是我的源代码。我想过滤“firstName”列,就像在 Facebook 的示例中一样(为简单起见)。错误源于调用 getSize() 时......但我怀疑问题出在其他地方。

class DataListWrapper 
    constructor(indexMap, data) 
        this._indexMap = indexMap;
        this._data = data;
    

    getSize() 
        return this._indexMap.length;
    

    getObjectAt(index) 
        return this._data.getObjectAt(
            this._indexMap[index],
        );
    


class NameTable extends React.Component 
    constructor(props) 
        super(props);

        this.testDataArr = []; // An array.
        this._dataList = this.testDataArr;

        console.log(JSON.stringify(this._dataList)); // It prints the array correctly.


        this.state = 
            filteredDataList: new DataListWrapper([], this._dataList)
        ;

        this._onFilterChange = this._onFilterChange.bind(this);
    

    _onFilterChange(e) 
        if (!e.target.value) 
            this.setState(
                filteredDataList: this._dataList,
            );
        

        var filterBy = e.target.value;
        var size = this._dataList.getSize();
        var filteredIndexes = [];
        for (var index = 0; index < size; index++) 
            var firstName = this._dataList.getObjectAt(index);
            if (firstName.indexOf(filterBy) !== -1) 
                filteredIndexes.push(index);
            
        

        this.setState(
            filteredDataList: new DataListWrapper(filteredIndexes, this._dataList),
        );
    

    render() 

        var filteredDataList = this.state.filteredDataList;


        if (!filteredDataList) 
            return <div>Loading table..  </div>;
        

        var rowsCount = filteredDataList.getSize();



        return (
            <div>
                <input onChange=this._onFilterChange type="text" placeholder='Search for first name.. ' />
                /*A table goes here, which renders fine normally without the search filter. */
            </div>
        );
    


export default NameTable

【问题讨论】:

很抱歉告诉你,但你做错了。首先,当您可以将对象存储在您的类中时,将其存储在状态中是非常糟糕的。您应该以您的状态存储:dataListLoaded。加载数据列表时,您必须设置 setState (dataListLoaded: true) 来重新渲染组件。 公平地说,我对 React 和一般前端的东西比较陌生。如果您可以提供您提到的内容的代码示例,我很乐意对其进行测试。 为了找到在 es6 中编码和反应的好方法,我看看这个:github.com/ryanmcdermott/clean-code-javascript 和一些像 material-ui 这样好的库的源代码 @cbll 你测试我的答案了吗?效果好吗? 【参考方案1】:

您的问题出在_onFilterChange 方法中。

你正在这样做:

var size = this._dataList.getSize();

this._dataList 只是一个数组,这就是该对象中不存在 getSize() 的原因。

如果我没有误解你应该这样做:

var size = this.state.filteredDataList.getSize();

循环内部也会发生同样的事情,你正在这样做:

var firstName = this._dataList.getObjectAt(index);

什么时候应该这样做:

var firstName = this.state.filteredDataList.getObjectAt(index);

您的 _onFilterChange 方法应如下所示:

_onFilterChange(e) 
    if (!e.target.value) 
      this.setState(
        filteredDataList: this._dataList,
      );
    

    var filterBy = e.target.value;
    //var size = this._dataList.getSize();
    var size = this.state.filteredDataList.getSize();
    var filteredIndexes = [];
    for (var index = 0; index < size; index++) 
      //var firstName = this._dataList.getObjectAt(index);
      var firstName = this.state.filteredDataList.getObjectAt(index);
      if (firstName.indexOf(filterBy) !== -1) 
        filteredIndexes.push(index);
      
    

    this.setState(
      filteredDataList: new DataListWrapper(filteredIndexes, this._dataList),
    );

【讨论】:

【参考方案2】:

getSize() 和 getObjectAt() 只能在实现这些方法的数据对象上调用,例如 DataListWrapper 对象。

如果您将纯数据数组传递给 render(),则它不提供 getSize() 和 getElementAt() 方法,并且对方法的调用将失败。

原始演示有效,因为 FakeObjectDataListStore 数据是一个对象(“FakeObjectDataListStore”),它实现了 getSize 和 getObjectAt 方法。

所以最简单的集成是确保传入的数据是提供这些方法的对象。根据我的“examples/FilterExample”案例,我发现最简单的集成(在与许多不好的整合后)是将现有的“helpers/FakeObjectDataListStore.js”变成我自己的 helpers/ObjectDataListStore.js(或选择你的名字)从而在整个设计中保留现有的方法包装结构和尺寸参数。然后,我简单地将对“假”组件的调用替换为对我自己的非包装本地列表行数组的引用。您可以将本地数据安排为静态数据,或从您使用的任何数据库环境动态加载。然后很容易修改 _setFiltered() 方法以过滤除 'firstName' 以外的其他内容。

FixedDataTable 最酷的地方在于它能够浏览大型列表, 并且开发人员可以编写自己的自定义单元格渲染器,例如在列表行的任何位置显示进度条、按钮或菜单。

【讨论】:

以上是关于反应固定数据表:未捕获类型错误:this._dataList.getSize 不是函数的主要内容,如果未能解决你的问题,请参考以下文章

反应和流星“未捕获的类型错误:无法读取未定义的属性'createElement'”

反应 - 未捕获的类型错误:无法读取未定义的属性“toLowerCase”

反应错误 - 未捕获的不变违规:元素类型无效

与 ES7 反应:未捕获的类型错误:无法读取未定义的属性“状态”[重复]

获取未捕获的类型错误:path.split 不是反应中的函数

未捕获的类型错误:无法设置未定义数据表的属性“_DT_CellIndex”