Skip to content

Commit

Permalink
feat(table): add getRowDescriptor property
Browse files Browse the repository at this point in the history
  • Loading branch information
GermanVor committed Jan 17, 2024
1 parent 252229e commit dc062af
Show file tree
Hide file tree
Showing 4 changed files with 85 additions and 6 deletions.
9 changes: 9 additions & 0 deletions src/components/Table/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ Additional functionality is enabled via HOCs:
| data | Data | `any[]` | |
| columns | Column parameters | `TableColumnConfig[]` | |
| verticalAlign | Vertical alignment of contents | `"top"` `"middle"` | |
| getRowDescriptor | Row mouseleave handler | `(item: any, index: number) => DescriptorType` | |
| getRowId | The row ID, used when selecting and sorting rows. If you skip a row, its ID will be the value of the field in the row data with the same name as the column ID | `string` `((item: any, index: number) => string)` | |
| getRowClassNames | Row CSS classes | `(item: any, index: number) => string[]` | |
| isRowDisabled | Condition for disabling columns | `(item: any, index: number) => boolean` | |
Expand All @@ -41,6 +42,14 @@ Additional functionality is enabled via HOCs:
| stickyHorizontalScroll | A horizontal sticky scroll in a table. NB: A table cannot have a fixed height and a sticky scroll at the same time. A sticky scroll will not work if the table has an overflow. | `boolean` | `false` |
| stickyHorizontalScrollBreakpoint | The threshold that the parent block should reach before making a scroll sticky. This is useful in the console, for example, when the groupActions bar closes the scroll. | `number` | `0` |

### DescriptorType

| Name | Description | Type | Default |
| :--------- | :------------------------------------------------------------------------------------------------------------------------------------------------------------- | :---------------------: | :---------: |
| id | The row ID, used when selecting and sorting rows. If you skip a row, its ID will be the value of the field in the row data with the same name as the column ID | `string` `undefined` | `undefined` |
| disabled | Column ID | `boolean` `undefined` | `undefined` |
| classNames | Row CSS classes | ` string[]` `undefined` | `undefined` |

### TableColumnConfig

| Name | Description | Type | Default |
Expand Down
58 changes: 53 additions & 5 deletions src/components/Table/Table.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,25 @@ export interface TableColumnConfig<I> {
meta?: Record<string, any>;
}

export interface DescriptorType {
/**
* Row ID.
* Used when selecting and sorting rows. If you pass a row,
* its ID will be the value of the field in the row data named the same as the column ID.
*/
id?: string;

/**
* Row CSS classes.
*/
classNames?: string[];

/**
* Condition for disabling columns.
*/
disabled?: boolean;
}

// TODO: Replace @default in props description with defaultProps in order to work with Storybook.
export interface TableProps<I> extends QAProps {
/** Data */
Expand All @@ -75,15 +94,32 @@ export interface TableProps<I> extends QAProps {
*/
stickyHorizontalScrollBreakpoint?: number;
/**
* @deprecated Use getRowDescriptor instead
*
* Row ID.
* Used when selecting and sorting rows. If you pass a row,
* its ID will be the value of the field in the row data named the same as the column ID.
*/
getRowId?: string | ((item: I, index: number) => string);
/** Row CSS classes. */
/**
* @deprecated Use getRowDescriptor instead
*
* Row CSS classes.
* */
getRowClassNames?: (item: I, index: number) => string[];
/** Condition for disabling columns. */
/**
* @deprecated Use getRowDescriptor instead
*
* Condition for disabling columns.
* */
isRowDisabled?: (item: I, index: number) => boolean;

/**
*
* @returns {DescriptorType} {@link DescriptorType}
*/
getRowDescriptor?: (item: I, index: number) => DescriptorType;

/** Row click handler. When passed row's hover is visible. */
onRowClick?: (item: I, index: number, event: React.MouseEvent<HTMLTableRowElement>) => void;
/** Row mouseenter handler. */
Expand Down Expand Up @@ -124,9 +160,15 @@ export class Table<I extends TableDataItem = Record<string, string>> extends Rea

// Static methods may be used by HOCs
static getRowId<I extends TableDataItem>(props: TableProps<I>, item: I, rowIndex?: number) {
const {data, getRowId} = props;
const {data, getRowId, getRowDescriptor} = props;
const index = rowIndex ?? data.indexOf(item);

const descriptor = getRowDescriptor?.(item, index);

if (descriptor?.id !== undefined) {
return descriptor.id;
}

if (typeof getRowId === 'function') {
return getRowId(item, index);
}
Expand Down Expand Up @@ -404,12 +446,18 @@ export class Table<I extends TableDataItem = Record<string, string>> extends Rea
verticalAlign,
edgePadding,
wordWrap,
getRowDescriptor,
} = this.props;
const {columnsStyles} = this.state;

const disabled = isRowDisabled ? isRowDisabled(item, rowIndex) : false;
const descriptor = getRowDescriptor?.(item, rowIndex);

const disabled = descriptor?.disabled || isRowDisabled?.(item, rowIndex) || false;

const additionalClassNames =
descriptor?.classNames || getRowClassNames?.(item, rowIndex) || [];

const interactive = Boolean(!disabled && onRowClick);
const additionalClassNames = getRowClassNames ? getRowClassNames(item, rowIndex) : [];

return (
<tr
Expand Down
4 changes: 3 additions & 1 deletion src/components/Table/__stories__/Table.stories.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,9 @@ export default {
},
} as Meta<TableProps<DataItem>>;

const DefaultTemplate: StoryFn<TableProps<DataItem>> = (args) => <Table {...args} />;
const DefaultTemplate: StoryFn<TableProps<DataItem>> = (args) => {
return <Table {...args} />;
};
export const Default = DefaultTemplate.bind({});

const EmptyDefaultTemplate: StoryFn<TableProps<DataItem>> = (args) => <Table {...args} />;
Expand Down
20 changes: 20 additions & 0 deletions src/components/Table/hoc/withTableSelection/withTableSelection.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,7 @@ export function withTableSelection<I extends TableDataItem, E extends {} = {}>(
columns,
onRowClick,
getRowClassNames,
getRowDescriptor,
...restTableProps
} = this.props;

Expand All @@ -50,6 +51,7 @@ export function withTableSelection<I extends TableDataItem, E extends {} = {}>(
columns={this.enhanceColumns(columns)}
onRowClick={this.enhanceOnRowClick(onRowClick)}
getRowClassNames={this.enhanceGetRowClassNames(getRowClassNames)}
getRowDescriptor={this.enhanceGetRowDescriptor(getRowDescriptor)}
/>
);
}
Expand Down Expand Up @@ -208,6 +210,24 @@ export function withTableSelection<I extends TableDataItem, E extends {} = {}>(
const classNames = getRowClassNames
? getRowClassNames(item, index).slice()
: [];

const id = Table.getRowId(this.props, item, index);
const selected = selectedIds.includes(id);

classNames.push(b('row', {selected}));

return classNames;
};
},
);

// eslint-disable-next-line @typescript-eslint/member-ordering
private enhanceGetRowDescriptor = _memoize(
(getRowDescriptor?: TableProps<I>['getRowDescriptor']) => {
return (item: I, index: number) => {
const {selectedIds} = this.props;
const classNames = getRowDescriptor?.(item, index).classNames?.slice() || [];

const id = Table.getRowId(this.props, item, index);
const selected = selectedIds.includes(id);

Expand Down

0 comments on commit dc062af

Please sign in to comment.