@justjarethb/table-core
Version:
Fork of @tanstack/table-core with support for event listeners in different windows
123 lines (101 loc) • 3.73 kB
text/typescript
import { Table, Row, RowModel, RowData } from '../types'
import { SortingFn } from '../features/Sorting'
import { memo } from '../utils'
export function getSortedRowModel<TData extends RowData>(): (
table: Table<TData>
) => () => RowModel<TData> {
return table =>
memo(
() => [table.getState().sorting, table.getPreSortedRowModel()],
(sorting, rowModel) => {
if (!rowModel.rows.length || !sorting?.length) {
return rowModel
}
const sortingState = table.getState().sorting
const sortedFlatRows: Row<TData>[] = []
// Filter out sortings that correspond to non existing columns
const availableSorting = sortingState.filter(sort =>
table.getColumn(sort.id)?.getCanSort()
)
const columnInfoById: Record<
string,
{
sortUndefined?: false | -1 | 1
invertSorting?: boolean
sortingFn: SortingFn<TData>
}
> = {}
availableSorting.forEach(sortEntry => {
const column = table.getColumn(sortEntry.id)
if (!column) return
columnInfoById[sortEntry.id] = {
sortUndefined: column.columnDef.sortUndefined,
invertSorting: column.columnDef.invertSorting,
sortingFn: column.getSortingFn(),
}
})
const sortData = (rows: Row<TData>[]) => {
// This will also perform a stable sorting using the row index
// if needed.
const sortedData = [...rows]
sortedData.sort((rowA, rowB) => {
for (let i = 0; i < availableSorting.length; i += 1) {
const sortEntry = availableSorting[i]!
const columnInfo = columnInfoById[sortEntry.id]!
const isDesc = sortEntry?.desc ?? false
let sortInt = 0
// All sorting ints should always return in ascending order
if (columnInfo.sortUndefined) {
const aValue = rowA.getValue(sortEntry.id)
const bValue = rowB.getValue(sortEntry.id)
const aUndefined = aValue === undefined
const bUndefined = bValue === undefined
if (aUndefined || bUndefined) {
sortInt =
aUndefined && bUndefined
? 0
: aUndefined
? columnInfo.sortUndefined
: -columnInfo.sortUndefined
}
}
if (sortInt === 0) {
sortInt = columnInfo.sortingFn(rowA, rowB, sortEntry.id)
}
// If sorting is non-zero, take care of desc and inversion
if (sortInt !== 0) {
if (isDesc) {
sortInt *= -1
}
if (columnInfo.invertSorting) {
sortInt *= -1
}
return sortInt
}
}
return rowA.index - rowB.index
})
// If there are sub-rows, sort them
sortedData.forEach(row => {
sortedFlatRows.push(row)
if (row.subRows?.length) {
row.subRows = sortData(row.subRows)
}
})
return sortedData
}
return {
rows: sortData(rowModel.rows),
flatRows: sortedFlatRows,
rowsById: rowModel.rowsById,
}
},
{
key: process.env.NODE_ENV === 'development' && 'getSortedRowModel',
debug: () => table.options.debugAll ?? table.options.debugTable,
onChange: () => {
table._autoResetPageIndex()
},
}
)
}