制表符在 adding/updating 行之后没有 re-sort 数据
Tabulator does not re-sort data after adding/updating row
我目前正在为 React 应用程序使用“tabulator-tables”版本 4.9.3。
此项目的要求之一是从 websocket 接收 live-stream 数据,其中包含要更新的现有行和要添加的新行的混合。
为了实现这一点,每当我从 websocket 收到一条新数据时,我都会检查 componentDidUpdate() 以查看 this.props.data 是否已更改,然后使用 tabulator 提供的 updateOrAddData() 函数相应地更新 table。
componentDidUpdate(prevProps) {
const { data } = this.props;
if (data !== prevProps.data) {
this.table.updateOrAddData([data]);
}
}
由于 table 填充了来自 websocket 的数据,我注意到没有调用 Tabulator 的嵌入式排序算法来确定添加到 [=46= 的新数据的正确排序位置] 通过 updateOrAddData()。如果用户没有明确点击某列header到re-sort这个数据,数据会根据用户提供的条件乱序
我试过的:
我尝试过的一个可能的解决方案是利用 updateOrAddData() 返回的承诺并手动调用 setSort()。
const { data } = this.props;
this.table.updateOrAddData(data).then(() => {
const sorters = this.table.getSorters();
this.table.setSort(sorters);
});
不幸的是,每当调用 setSort() 时,滚动条 returns 到 table 的顶部,因此由于接收到新数据而难以滚动整个 table大约每秒。
为了演示这个问题,我用每秒随机生成数据的 setInterval() 函数替换了 websocket 连接。
我目前使用的代码可以在 https://codesandbox.io/s/currying-brook-ck8r3 或以下访问。
代码:
import React, { Component, createRef, useEffect, useState } from "react";
import Tabulator from "tabulator-tables";
import "../node_modules/tabulator-tables/dist/css/tabulator.min.css";
class TabulatorTable extends Component {
constructor(props) {
super(props);
this.el = createRef();
this.table = null;
}
componentDidMount() {
const { columns, options } = this.props;
this.table = new Tabulator(this.el, {
columns,
data: [],
...options
});
}
componentDidUpdate(prevProps) {
const { data } = this.props;
if (data !== prevProps.data) {
this.table.updateOrAddData([data]);
}
}
render() {
return <div ref={(el) => (this.el = el)} />;
}
}
const App = () => {
const [mockWsData, setMockWsData] = useState({});
const columns = [
{
field: "id",
title: "ID",
sorter: "number"
},
{
field: "letters",
title: "Random letters",
sorter: "string"
}
];
const options = {
height: "500px",
index: "id",
layout: "fitColumns",
reactiveData: true
};
useEffect(() => {
const generateFakeData = setInterval(() => {
setMockWsData({
id: Math.floor(Math.random() * Math.floor(15)),
letters: Math.random().toString(36).substring(2, 7)
});
}, 500);
return () => clearInterval(generateFakeData);
}, [setMockWsData]);
return (
<TabulatorTable columns={columns} data={mockWsData} options={options} />
);
};
export default App;
我决定扩展利用 updateOrAddData() 返回的承诺并手动设置 scrollTop 的方法。
currentTopRow() {
const { rowManager } = this.table;
const { scrollTop } = rowManager.element;
let topRow = false;
let topOffset = false;
const rows = rowManager.getDisplayRows();
for (let i = rowManager.vDomTop; i <= rowManager.vDomBottom; i += 1) {
const row = rows[i];
if (row) {
const diff = scrollTop - row.getElement().offsetTop;
if (topOffset === false || Math.abs(diff) < topOffset) {
topOffset = diff;
topRow = row.getComponent().getIndex();
} else break;
}
}
return topRow;
}
componentDidUpdate(prevProps) {
const { data } = this.props;
if (data !== prevProps.data) {
const topRow = this.currentTopRow();
this.table.updateOrAddData([data]).then(() => {
this.table.setSort(this.table.getSorters());
if (topRow) {
this.table.scrollToRow(topRow, 'top', true);
}
});
}
}
我目前正在为 React 应用程序使用“tabulator-tables”版本 4.9.3。
此项目的要求之一是从 websocket 接收 live-stream 数据,其中包含要更新的现有行和要添加的新行的混合。
为了实现这一点,每当我从 websocket 收到一条新数据时,我都会检查 componentDidUpdate() 以查看 this.props.data 是否已更改,然后使用 tabulator 提供的 updateOrAddData() 函数相应地更新 table。
componentDidUpdate(prevProps) {
const { data } = this.props;
if (data !== prevProps.data) {
this.table.updateOrAddData([data]);
}
}
由于 table 填充了来自 websocket 的数据,我注意到没有调用 Tabulator 的嵌入式排序算法来确定添加到 [=46= 的新数据的正确排序位置] 通过 updateOrAddData()。如果用户没有明确点击某列header到re-sort这个数据,数据会根据用户提供的条件乱序
我试过的:
我尝试过的一个可能的解决方案是利用 updateOrAddData() 返回的承诺并手动调用 setSort()。
const { data } = this.props;
this.table.updateOrAddData(data).then(() => {
const sorters = this.table.getSorters();
this.table.setSort(sorters);
});
不幸的是,每当调用 setSort() 时,滚动条 returns 到 table 的顶部,因此由于接收到新数据而难以滚动整个 table大约每秒。
为了演示这个问题,我用每秒随机生成数据的 setInterval() 函数替换了 websocket 连接。
我目前使用的代码可以在 https://codesandbox.io/s/currying-brook-ck8r3 或以下访问。
代码:
import React, { Component, createRef, useEffect, useState } from "react";
import Tabulator from "tabulator-tables";
import "../node_modules/tabulator-tables/dist/css/tabulator.min.css";
class TabulatorTable extends Component {
constructor(props) {
super(props);
this.el = createRef();
this.table = null;
}
componentDidMount() {
const { columns, options } = this.props;
this.table = new Tabulator(this.el, {
columns,
data: [],
...options
});
}
componentDidUpdate(prevProps) {
const { data } = this.props;
if (data !== prevProps.data) {
this.table.updateOrAddData([data]);
}
}
render() {
return <div ref={(el) => (this.el = el)} />;
}
}
const App = () => {
const [mockWsData, setMockWsData] = useState({});
const columns = [
{
field: "id",
title: "ID",
sorter: "number"
},
{
field: "letters",
title: "Random letters",
sorter: "string"
}
];
const options = {
height: "500px",
index: "id",
layout: "fitColumns",
reactiveData: true
};
useEffect(() => {
const generateFakeData = setInterval(() => {
setMockWsData({
id: Math.floor(Math.random() * Math.floor(15)),
letters: Math.random().toString(36).substring(2, 7)
});
}, 500);
return () => clearInterval(generateFakeData);
}, [setMockWsData]);
return (
<TabulatorTable columns={columns} data={mockWsData} options={options} />
);
};
export default App;
我决定扩展利用 updateOrAddData() 返回的承诺并手动设置 scrollTop 的方法。
currentTopRow() {
const { rowManager } = this.table;
const { scrollTop } = rowManager.element;
let topRow = false;
let topOffset = false;
const rows = rowManager.getDisplayRows();
for (let i = rowManager.vDomTop; i <= rowManager.vDomBottom; i += 1) {
const row = rows[i];
if (row) {
const diff = scrollTop - row.getElement().offsetTop;
if (topOffset === false || Math.abs(diff) < topOffset) {
topOffset = diff;
topRow = row.getComponent().getIndex();
} else break;
}
}
return topRow;
}
componentDidUpdate(prevProps) {
const { data } = this.props;
if (data !== prevProps.data) {
const topRow = this.currentTopRow();
this.table.updateOrAddData([data]).then(() => {
this.table.setSort(this.table.getSorters());
if (topRow) {
this.table.scrollToRow(topRow, 'top', true);
}
});
}
}