1
0
mirror of https://github.com/lensapp/lens.git synced 2025-05-20 05:10:56 +00:00

add more typing to sorting of tables

Signed-off-by: Sebastian Malton <sebastian@malton.name>
This commit is contained in:
Sebastian Malton 2020-12-15 14:41:24 -05:00
parent 0015770739
commit 0a5b8b37db
3 changed files with 54 additions and 18 deletions

View File

@ -0,0 +1,3 @@
export function hasKey<K extends string, V>(obj: Record<K, V>, testKey: string): testKey is K {
return obj.hasOwnProperty(testKey);
}

View File

@ -17,4 +17,5 @@ export * from "./openExternal";
export * from "./rectify-array"; export * from "./rectify-array";
export * from "./downloadFile"; export * from "./downloadFile";
export * from "./escapeRegExp"; export * from "./escapeRegExp";
export * from "./has-key";
export * from "./tar"; export * from "./tar";

View File

@ -3,7 +3,7 @@ import "./table.scss";
import React from "react"; import React from "react";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import { computed, observable } from "mobx"; import { computed, observable } from "mobx";
import { autobind, cssNames, noop } from "../../utils"; import { autobind, cssNames, hasKey, noop } from "../../utils";
import { TableRow, TableRowElem, TableRowProps } from "./table-row"; import { TableRow, TableRowElem, TableRowProps } from "./table-row";
import { TableHead, TableHeadElem, TableHeadProps } from "./table-head"; import { TableHead, TableHeadElem, TableHeadProps } from "./table-head";
import { TableCellElem } from "./table-cell"; import { TableCellElem } from "./table-cell";
@ -14,7 +14,10 @@ import { ItemObject } from "../../item.store";
// todo: refactor + decouple search from location // todo: refactor + decouple search from location
export type TableOrderBy = "asc" | "desc" | string; export enum TableOrderBy {
ASCENDING = "asc",
DECENDING = "desc",
}
export type TableSortParams<SortingOption extends string> = { sortBy: SortingOption; orderBy: TableOrderBy }; export type TableSortParams<SortingOption extends string> = { sortBy: SortingOption; orderBy: TableOrderBy };
export type TableSortCallback<D> = (data: D) => string | number | (string | number)[]; export type TableSortCallback<D> = (data: D) => string | number | (string | number)[];
@ -85,12 +88,42 @@ export interface TableProps<Entry extends ItemObject, SortingOption extends stri
*/ */
virtual?: boolean; virtual?: boolean;
/**
* the number of pixels to render between subsequent rows
*/
rowPadding?: string; rowPadding?: string;
/**
* the number of pixels to render for a specific row
*/
rowLineHeight?: string; rowLineHeight?: string;
customRowHeights?: (item: object, lineHeight: number, paddings: number) => number;
/**
* A function for generating a special number for a specific item
*/
customRowHeights?: (item: any, lineHeight: number, paddings: number) => number;
/**
*
*/
getTableRow?: (uid: string) => React.ReactElement<TableRowProps>; getTableRow?: (uid: string) => React.ReactElement<TableRowProps>;
} }
function validateOrderBy(source: string | null, defaultDirection = TableOrderBy.DECENDING): TableOrderBy {
const transformed = source?.toLowerCase();
if (transformed === TableOrderBy.ASCENDING || transformed === TableOrderBy.DECENDING) {
return transformed;
}
return defaultDirection;
}
const swapOrderBy: Record<TableOrderBy, TableOrderBy> = {
[TableOrderBy.ASCENDING]: TableOrderBy.DECENDING,
[TableOrderBy.DECENDING]: TableOrderBy.ASCENDING,
};
@observer @observer
export class Table<Entry extends ItemObject = ItemObject, SortingOption extends string = string> extends React.Component<TableProps<Entry, SortingOption>> { export class Table<Entry extends ItemObject = ItemObject, SortingOption extends string = string> extends React.Component<TableProps<Entry, SortingOption>> {
static defaultProps: TableProps<ItemObject, string> = { static defaultProps: TableProps<ItemObject, string> = {
@ -104,12 +137,16 @@ export class Table<Entry extends ItemObject = ItemObject, SortingOption extends
@observable sortParamsLocal = this.props.sortByDefault; @observable sortParamsLocal = this.props.sortByDefault;
@computed get sortParams(): Partial<TableSortParams<SortingOption>> { @computed get sortParams(): Partial<TableSortParams<SortingOption>> {
if (this.props.sortSyncWithUrl) { const { sortSyncWithUrl, sortable } = this.props;
const sortBy = navigation.searchParams.get("sortBy") as SortingOption;
const orderBy = navigation.searchParams.get("orderBy");
if (sortSyncWithUrl) {
const sortBy = navigation.searchParams.get("sortBy");
const orderBy = validateOrderBy(navigation.searchParams.get("orderBy"));
if (sortable && hasKey(sortable, sortBy)) {
return { sortBy, orderBy }; return { sortBy, orderBy };
} }
}
return this.sortParamsLocal || {}; return this.sortParamsLocal || {};
} }
@ -155,7 +192,7 @@ export class Table<Entry extends ItemObject = ItemObject, SortingOption extends
return orderBy( return orderBy(
items, items,
sortingCallback, sortingCallback,
sortParams.orderBy as any sortParams.orderBy
); );
} }
@ -165,25 +202,20 @@ export class Table<Entry extends ItemObject = ItemObject, SortingOption extends
if (sortSyncWithUrl) { if (sortSyncWithUrl) {
setQueryParams(params); setQueryParams(params);
} } else {
else {
this.sortParamsLocal = params; this.sortParamsLocal = params;
} }
if (onSort) { onSort?.(params);
onSort(params);
}
} }
@autobind() @autobind()
sort(colName: SortingOption) { sort(colName: SortingOption) {
const { sortBy, orderBy } = this.sortParams; const { sortBy, orderBy = TableOrderBy.ASCENDING } = this.sortParams;
const sameColumn = sortBy == colName; const newOrderBy = (sortBy === colName) ? swapOrderBy[orderBy] : TableOrderBy.DECENDING;
const newSortBy: SortingOption = colName;
const newOrderBy: TableOrderBy = (!orderBy || !sameColumn || orderBy === "desc") ? "asc" : "desc";
this.onSort({ this.onSort({
sortBy: newSortBy, sortBy: colName,
orderBy: newOrderBy, orderBy: newOrderBy,
}); });
} }