Skip to content

Commit

Permalink
doc: add doc afterRow
Browse files Browse the repository at this point in the history
  • Loading branch information
santiago.trigo committed Sep 4, 2024
1 parent c4ef013 commit 470178b
Show file tree
Hide file tree
Showing 3 changed files with 198 additions and 45 deletions.
2 changes: 1 addition & 1 deletion packages/table/src/hooks/useAfterRow/useAfterRow.ts
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ export const useRenderAfterRow = ({
onRowDefsChange: (rowDefs: TRowDef[]) => void;
colDefs: TColDef[];
}) => {
const [selection, setSelection] = React.useState(initialSelection);
const [selection, setSelection] = React.useState(initialSelection || []);

const colDefs = [
{
Expand Down
110 changes: 110 additions & 0 deletions packages/table/stories/AfterRow.mdx
Original file line number Diff line number Diff line change
@@ -0,0 +1,110 @@
import { Meta, Source, ArgTypes, Canvas } from '@storybook/blocks';

import * as Stories from './AfterRow.stories';

<Meta of={Stories} />

# AfterRow

After row is a functionality to expand a row to add more information to that row.

<Canvas of={Stories.Basic} />

For functionality we have two options:

- Start from the **afterRow rendered in the dom** but hidden (top example). This case may impair the initial loading of the table, but will favor a smoother use by the user. It must be remembered that the virtualized table does not render all the elements.
- The **afterRow is created and deleted as you click on the corresponding row** (example below). If the afterRow renders some very heavy component this option can be good, since it relegates the load until the user decides that he wants to see that content.

> Both options have their pros and cons, we advise to use one or the other depending on the use case.
<Canvas of={Stories.BasicNoRenderDom} />

## AfterRow rendered in the dom (useRenderAfterRow).

### Build dataset

We must add to the data set the rows corresponding to the afterrow we are going to represent. For it we can use the function **addAfterRowsToData** to which we pass the data set and it will return us a new data set with after row. This function will also return an array with the ids of the new rows. This will facilitate the construction of the row definition that we will see later.

```ts
const [dataWithAfterRows, afterRowIds] = addAfterRowsToData(dataOrdered);
```

### Build row definition

We can start with an empty row definition or with a row already defined. But we need to add the definition of the new after row in case they are not defined. For this we have the function **addAfterRowsToRowDefs**.

This function has 4 parameters:

- **initalRowDefs**: the initial configuration of rows, always in array format and with an id that allows us to identify it.
- **afterRowIds**: string array with the ids of the new rows we added in the previous point.
- **afterRowRenderer**: here we can define the cellrenderer that will be represented in the afterRow. This cell renderer receives the information of the row to which it belongs so that it can make use of the data (it can be seen in the previous examples) or any react component can be represented without making use of these data.
- **afterRowHeight**: in case we want a custom height for after row in general we can define it here.

```ts
const [rowDefs, setRowDefs] = React.useState(
addAfterRowsToRowDefs(
initalRowDefs,
afterRowIds,
afterRowRenderer,
afterRowHeight,
),
);
```

### useRenderAfterRow

We have prepared a hook that allows us to control the behavior of showing or hiding afterRow. This hook will add column definition and click behavior.

This function has 4 parameters:

- **rowDefs**: Definition of rows built in the previous point.
- **initialSelection**: if we want an after row to be displayed from the beginning we can do it by passing the id of the row.
- **onRowDefsChange**: callback that allows us to update the row definition change as needed.
- **colDefs**: definition of initial columns.

Initially this function will return the modified column definition and when clicked will update the row definition.

```ts
const { colDefs } = useRenderAfterRow({
rowDefs,
initialSelection,
onRowDefsChange: (newRowDefs) => {
setRowDefs(newRowDefs);
},
colDefs: colDefsInitial,
});
```

## AfterRow is created and deleted as you click on the corresponding row (useOnDemandAfterRow).

Unlike the previous option, here we do not need to modify previously neither the data nor the column definitions. Everything is relegated to the hook where all the necessary modifications will be made.

### useOnDemandAfterRow

This hook will allow us to update the data, define columns and define rows. Initially there are no after row rows in the dom. When you click on a row it creates or deletes the corresponding row in the data and modifies the row definition.

This function has 7 parameters:

- **rowDefs**: Definition of rows built in the previous point.
- **onRowDefsChange**: callback that allows us to update the row definition change as needed.
- **colDefs**: definition of initial columns.
- **data**: dataset.
- **onDataChange**: callback that allows us to update the data change as needed.
- **afterRowRenderer**: here we can define the cellrenderer that will be represented in the afterRow. This cell renderer receives the information of the row to which it belongs so that it can make use of the data (it can be seen in the previous examples) or any react component can be represented without making use of these data.
- **afterRowHeight**: in case we want a custom height for after row in general we can define it here.

```ts
const { colDefs } = useOnDemandAfterRow({
rowDefs,
onRowDefsChange: (newRowDefs) => {
setRowDefs(newRowDefs);
},
colDefs: colDefsInitial,
data,
onDataChange: (newData) => {
setData(newData);
},
afterRowRenderer,
afterRowHeight,
});
```
131 changes: 87 additions & 44 deletions packages/table/stories/AfterRow.stories.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -16,10 +16,10 @@ import {

import { ROW_HEIGHT_MD } from '../src/constants';
import { Flex } from '@devoinc/genesys-ui';
import { useAfterRow } from '../src/hooks';
import { useOnDemandAfterRow, useRenderAfterRow } from '../src/hooks';

const meta: Meta<typeof Table> = {
title: 'Components/Layout/Table/AfterRow/Basic',
title: 'Components/Layout/Table/AfterRow',
component: Table,
parameters: {
layout: 'fullscreen',
Expand All @@ -38,59 +38,72 @@ const initialData = Holo.of()
booleanValue: 'bool',
name: 'name',
})
.repeat(20)
.repeat(5)
.generate();

const colDefsInitial = [
{
id: 'id',
preset: 'text',
headerName: 'ID',
cellRenderer: TextRenderer,
sortable: true,
},
{
id: 'booleanValue',
headerName: 'Boolean value',
preset: 'boolean',
editable: true,
cellRenderer: BooleanRenderer,
},
{
id: 'name',
headerName: 'Name',
preset: 'text',
editable: true,
cellRenderer: TextRenderer,
sortable: true,
},
];

const initalRowDefs = [
{
id: `afterRow-2`,
hide: false,
},
];

const initialSelection = ['2']

const BasicCmp = ({ afterRowRenderer, afterRowHeight }) => {
const { orderStruct, onSort } = useOrderStruct([
{ id: 'text', sort: 'desc' },
]);

const dataOrdered = [...initialData].sort(
orderDataByOrderStruct(orderStruct),
);

const [dataWithAfterRows, afterRowIds] = addAfterRowsToData(dataOrdered);

const colDefsInitial = [
{
id: 'id',
preset: 'text',
headerName: 'ID',
cellRenderer: TextRenderer,
sortable: true,
},
{
id: 'booleanValue',
headerName: 'Boolean value',
preset: 'boolean',
editable: true,
cellRenderer: BooleanRenderer,
},
{
id: 'name',
headerName: 'Name',
preset: 'text',
editable: true,
cellRenderer: TextRenderer,
sortable: true,
},
];

const [rowDefs, setRowDefs] = React.useState(
addAfterRowsToRowDefs(afterRowIds, afterRowRenderer, afterRowHeight),
addAfterRowsToRowDefs(
initalRowDefs,
afterRowIds,
afterRowRenderer,
afterRowHeight,
),
);

const { colDefs } = useAfterRow({
const { colDefs } = useRenderAfterRow({
rowDefs,
initialSelection,
onRowDefsChange: (newRowDefs) => {
setRowDefs(newRowDefs);
},
colDefs: colDefsInitial,
});

return (
<Flex flexDirection="column" gap="cmp-md">
<Flex flexDirection="column" gap="cmp-md" height={'auto'}>
<Flex.Item>
<Table
data={dataWithAfterRows}
Expand All @@ -110,6 +123,43 @@ const BasicCmp = ({ afterRowRenderer, afterRowHeight }) => {
);
};

const BasicCmpNoRenderAfterRow = ({ afterRowRenderer, afterRowHeight }) => {
const [data, setData] = React.useState(initialData);

const [rowDefs, setRowDefs] = React.useState([]);

const { colDefs } = useOnDemandAfterRow({
rowDefs,
onRowDefsChange: (newRowDefs) => {
setRowDefs(newRowDefs);
},
colDefs: colDefsInitial,
data,
onDataChange: (newData) => {
setData(newData);
},
afterRowRenderer,
afterRowHeight,
});

return (
<Flex flexDirection="column" gap="cmp-md" height={'auto'}>
<Flex.Item>
<Table
data={data}
colDefs={colDefs}
rowDefs={rowDefs}
maxHeight={'80vh'}
rowHeight={ROW_HEIGHT_MD}
resizableColumns={true}
highlightColumnsOnHover={true}
showFilters={false}
/>
</Flex.Item>
</Flex>
);
};

export const Basic: Story = {
render: () => (
<BasicCmp
Expand All @@ -119,18 +169,11 @@ export const Basic: Story = {
),
};

export const RenderImg: Story = {
export const BasicNoRenderDom: Story = {
render: () => (
<BasicCmp
afterRowHeight={200}
afterRowRenderer={() => (
<div>
<img
src=""
alt="Dinosaur"
/>
</div>
)}
<BasicCmpNoRenderAfterRow
afterRowRenderer={({ row }) => row.name as string}
afterRowHeight={36}
/>
),
};

0 comments on commit 470178b

Please sign in to comment.