Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[XGrid] Add onViewportRowsChange prop #2038

Merged
merged 46 commits into from
Aug 11, 2021
Merged
Show file tree
Hide file tree
Changes from 28 commits
Commits
Show all changes
46 commits
Select commit Hold shift + click to select a range
97f4d34
Add onVirtualPageChange option
DanailH Jul 5, 2021
4127f39
PR comments
DanailH Jul 6, 2021
863378b
test
DanailH Jul 6, 2021
5afc0da
Resolve conflicts
DanailH Jul 6, 2021
3518322
Renameing
DanailH Jul 8, 2021
83c6d02
resolve conflicts
DanailH Jul 26, 2021
551cb17
Rework onVirtualRowsChange
DanailH Jul 27, 2021
9d7f44e
PR comments
DanailH Jul 27, 2021
36fca97
fix docs
DanailH Jul 27, 2021
315b58e
Merge branch 'master' of github.com:mui-org/material-ui-x into featur…
DanailH Jul 27, 2021
3ba0e5c
Trigger notification
DanailH Jul 27, 2021
cc2a479
PR comments
DanailH Jul 28, 2021
664c45f
Trigger Build
DanailH Jul 28, 2021
4bca503
Story
DanailH Jul 28, 2021
96bfd37
Trigger Build
DanailH Jul 28, 2021
15460f4
Revert story changes
DanailH Jul 30, 2021
d6d4845
Update implementation
DanailH Aug 3, 2021
f94bfec
Update docs/pages/api-docs/data-grid/x-grid.md
DanailH Aug 5, 2021
a245996
Update packages/grid/_modules_/grid/constants/eventsConstants.ts
DanailH Aug 5, 2021
4889fc9
Update packages/grid/_modules_/grid/models/gridOptions.tsx
DanailH Aug 5, 2021
0a3f5f7
adjust tests
DanailH Aug 5, 2021
3477c89
Merge branch 'feature/DataGrid-1715-onVirtualPageChange' of github.co…
DanailH Aug 5, 2021
f90ab87
update docs
DanailH Aug 5, 2021
0edf6ae
Update packages/grid/x-grid/src/tests/events.XGrid.test.tsx
DanailH Aug 5, 2021
b72127c
Update packages/grid/x-grid/src/tests/events.XGrid.test.tsx
DanailH Aug 5, 2021
191848f
Update packages/grid/x-grid/src/tests/events.XGrid.test.tsx
DanailH Aug 5, 2021
2c83269
Resolve conflicts
DanailH Aug 5, 2021
05e6749
Fixing tests
DanailH Aug 5, 2021
7f77ba5
PR comments
DanailH Aug 9, 2021
db1dc5c
Conflicts
DanailH Aug 9, 2021
d301dc8
resolve conflicts
DanailH Aug 10, 2021
c525e4a
Merge branch 'master' of github.com:mui-org/material-ui-x into featur…
DanailH Aug 10, 2021
70c1587
Fix tests
DanailH Aug 10, 2021
72364ca
Fix test
DanailH Aug 10, 2021
fd24d6e
test
DanailH Aug 10, 2021
d9c4bf5
Tests
DanailH Aug 10, 2021
ce57a0e
Resolve conflicts
DanailH Aug 11, 2021
beeaed6
docs update
DanailH Aug 11, 2021
e16832f
Fix windows tests
DanailH Aug 11, 2021
bfe9ac0
Fix tests under Windows
DanailH Aug 11, 2021
30dac03
Update packages/grid/_modules_/grid/constants/eventsConstants.ts
DanailH Aug 11, 2021
b74b946
Update packages/grid/_modules_/grid/models/gridOptions.tsx
DanailH Aug 11, 2021
7715ee4
PR comments
DanailH Aug 11, 2021
219085c
Merge branch 'feature/DataGrid-1715-onVirtualPageChange' of github.co…
DanailH Aug 11, 2021
5bf84c1
formatting and docs
DanailH Aug 11, 2021
2959a5b
small typo
m4theushw Aug 11, 2021
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions docs/pages/api-docs/data-grid/x-grid.md
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,7 @@ import { XGrid } from '@material-ui/x-grid';
| <span class="prop-name">onRowsScrollEnd</span> | <span class="prop-type">(param: GridRowScrollEndParams, event: MuiEvent<{}>) => void</span> | | Callback fired when scrolling to the bottom of the grid viewport. |
| <span class="prop-name">onSelectionModelChange</span> | <span class="prop-type">(model: GridSelectionModel) => void</span> | | Callback fired when the selection state of one or multiple rows changes. |
| <span class="prop-name">onSortModelChange</span> | <span class="prop-type">(model: GridSortModel) => void</span> | | Callback fired when the sort model changes before a column is sorted. |
| <span class="prop-name">onViewportRowsChange</span> | <span class="prop-type">(params: GridViewportRowsChangeParams) => void</span> | | Callback fired when the rows in the viewport change. |
| <span class="prop-name">page</span> | <span class="prop-type">number</span> | 1 | Set the current page. |
| <span class="prop-name">pageSize</span> | <span class="prop-type">number</span> | 100 | Set the number of rows in one page. |
| <span class="prop-name">pagination</span> | <span class="prop-type">boolean</span> | false | If `true`, pagination is enabled. |
Expand Down
4 changes: 4 additions & 0 deletions docs/src/pages/components/data-grid/events/events.json
Original file line number Diff line number Diff line change
Expand Up @@ -163,5 +163,9 @@
{
"name": "columnVisibilityChange",
"description": "Fired when a column visibility changes. Called with a GridColumnVisibilityChangeParams object."
},
{
"name": "viewportRowsChange",
"description": "Fired when the rows in the viewport change. Called with a GridViewportRowsChange object."
}
]
6 changes: 6 additions & 0 deletions packages/grid/_modules_/grid/constants/eventsConstants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -441,3 +441,9 @@ export const GRID_STATE_CHANGE = 'stateChange';
* @event
*/
export const GRID_COLUMN_VISIBILITY_CHANGE = 'columnVisibilityChange';

/**
* Fired when the rows in the viewport change. Called with a [[GridViewportRowsChange]] object.
* @event
*/
export const GRID_VIEWPORT_ROWS_CHANGE = 'viewportRowsChange';
Original file line number Diff line number Diff line change
Expand Up @@ -2,50 +2,103 @@ import * as React from 'react';
import { optionsSelector } from '../../utils/optionsSelector';
import { GridApiRef } from '../../../models/api/gridApiRef';
import { useGridSelector } from '../core/useGridSelector';
import { GRID_ROWS_SCROLL, GRID_ROWS_SCROLL_END } from '../../../constants/eventsConstants';
import {
GRID_ROWS_SCROLL,
GRID_ROWS_SCROLL_END,
GRID_VIEWPORT_ROWS_CHANGE,
} from '../../../constants/eventsConstants';
import { gridContainerSizesSelector } from '../../root/gridContainerSizesSelector';
import { useGridApiEventHandler, useGridApiOptionHandler } from '../../root/useGridApiEventHandler';
import { GridRowScrollEndParams } from '../../../models/params/gridRowScrollEndParams';
import { visibleGridColumnsSelector } from '../columns/gridColumnsSelector';
import { GridComponentProps } from '../../../GridComponentProps';
import { renderStateSelector } from '../virtualization/renderingStateSelector';
import { GridViewportRowsChangeParams } from '../../../models/params/gridViewportRowsChangeParams';
import { GridScrollParams } from '../../../models/params/gridScrollParams';

export const useGridInfiniteLoader = (
apiRef: GridApiRef,
props: Pick<GridComponentProps, 'onRowsScrollEnd'>,
props: Pick<GridComponentProps, 'onRowsScrollEnd' | 'onViewportRowsChange'>,
): void => {
const options = useGridSelector(apiRef, optionsSelector);
const containerSizes = useGridSelector(apiRef, gridContainerSizesSelector);
const visibleColumns = useGridSelector(apiRef, visibleGridColumnsSelector);
const isInScrollBottomArea = React.useRef<boolean>(false);
const renderState = useGridSelector(apiRef, renderStateSelector);
const previousRenderContext = React.useRef<null | {
firstRowIndex: number;
lastRowIndex: number;
}>(null);

const handleRowsScrollEnd = React.useCallback(
(scrollPosition: GridScrollParams) => {
if (!containerSizes) {
return;
}

const scrollPositionBottom =
scrollPosition.top + containerSizes.windowSizes.height + options.scrollEndThreshold;

if (scrollPositionBottom < containerSizes.dataContainerSizes.height) {
isInScrollBottomArea.current = false;
}

if (
scrollPositionBottom >= containerSizes.dataContainerSizes.height &&
!isInScrollBottomArea.current
) {
const rowScrollEndParam: GridRowScrollEndParams = {
api: apiRef,
visibleColumns,
viewportPageSize: containerSizes.viewportPageSize,
virtualRowsCount: containerSizes.virtualRowsCount,
};
apiRef.current.publishEvent(GRID_ROWS_SCROLL_END, rowScrollEndParam);
isInScrollBottomArea.current = true;
}
},
[apiRef, options, visibleColumns, containerSizes],
);

const handleGridScroll = React.useCallback(() => {
if (!containerSizes) {
if (!props.onRowsScrollEnd && !props.onViewportRowsChange) {
return;
}
DanailH marked this conversation as resolved.
Show resolved Hide resolved

const scrollPosition = apiRef.current.getScrollPosition();
const scrollPositionBottom =
scrollPosition.top + containerSizes.windowSizes.height + options.scrollEndThreshold;

if (scrollPositionBottom < containerSizes.dataContainerSizes.height) {
isInScrollBottomArea.current = false;
if (props.onRowsScrollEnd) {
handleRowsScrollEnd(scrollPosition);
}
DanailH marked this conversation as resolved.
Show resolved Hide resolved
}, [props, apiRef, handleRowsScrollEnd]);

// TODO: Check if onViewportRowsChange works as expected once virtualization is reworked
React.useEffect(() => {
const renderContext = renderState.renderContext!;

if (!props.onViewportRowsChange || !renderContext) {
DanailH marked this conversation as resolved.
Show resolved Hide resolved
return;
}

if (
scrollPositionBottom >= containerSizes.dataContainerSizes.height &&
!isInScrollBottomArea.current
!previousRenderContext.current ||
renderContext.firstRowIdx !== previousRenderContext.current.firstRowIndex ||
renderContext.lastRowIdx !== previousRenderContext.current.lastRowIndex
) {
const rowScrollEndParam: GridRowScrollEndParams = {
api: apiRef,
visibleColumns,
viewportPageSize: containerSizes.viewportPageSize,
virtualRowsCount: containerSizes.virtualRowsCount,
const viewportRowsChangeParams: GridViewportRowsChangeParams = {
firstRowIndex: renderContext.firstRowIdx!,
lastRowIndex: renderContext.lastRowIdx!,
};
apiRef.current.publishEvent(GRID_ROWS_SCROLL_END, rowScrollEndParam);
isInScrollBottomArea.current = true;
apiRef.current.publishEvent(GRID_VIEWPORT_ROWS_CHANGE, viewportRowsChangeParams);
}
}, [options, containerSizes, apiRef, visibleColumns]);

previousRenderContext.current = {
firstRowIndex: renderContext.firstRowIdx!,
lastRowIndex: renderContext.lastRowIdx!,
};
}, [apiRef, props.onViewportRowsChange, renderState]);

useGridApiEventHandler(apiRef, GRID_ROWS_SCROLL, handleGridScroll);
useGridApiOptionHandler(apiRef, GRID_ROWS_SCROLL_END, props.onRowsScrollEnd);
useGridApiOptionHandler(apiRef, GRID_VIEWPORT_ROWS_CHANGE, props.onViewportRowsChange);
};
5 changes: 5 additions & 0 deletions packages/grid/_modules_/grid/models/gridOptions.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ import { GridResizeParams } from './params/gridResizeParams';
import { GridColumnResizeParams } from './params/gridColumnResizeParams';
import { GridColumnVisibilityChangeParams } from './params/gridColumnVisibilityChangeParams';
import { GridClasses } from './gridClasses';
import { GridViewportRowsChangeParams } from './params/gridViewportRowsChangeParams';

export type MuiEvent<E> = E & {
defaultMuiPrevented?: boolean;
Expand Down Expand Up @@ -534,6 +535,10 @@ export interface GridOptions {
* @param {MuiCallbackDetails} details Additional details for this callback.
*/
onStateChange?: (params: any, event: MuiEvent<{}>, details?: any) => void;
/**
* Callback fired when the rows in the viewport change.
*/
onViewportRowsChange?: (params: GridViewportRowsChangeParams) => void;
DanailH marked this conversation as resolved.
Show resolved Hide resolved
/**
* Set the current page.
* @default 1
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
/**
* Object passed as parameter of the virtual rows change event.
*/
export interface GridViewportRowsChangeParams {
/**
* The index of the first row in the viewport.
*/
firstRowIndex: number;
/**
* The index of the last row in the viewport.
*/
lastRowIndex: number;
}
13 changes: 13 additions & 0 deletions packages/grid/data-grid/src/DataGridPropTypes.ts
Original file line number Diff line number Diff line change
Expand Up @@ -134,6 +134,19 @@ export const DATA_GRID_PROPTYPES = {
}
return null;
}),
onViewportRowsChange: chainPropTypes(PropTypes.any, (props: any) => {
if (props.onViewportRowsChange != null) {
return new Error(
[
`Material-UI: \`onViewportRowsChange\` is not a valid prop.`,
'onViewportRowsChange is not available in the MIT version.',
'',
'You need to upgrade to the XGrid component to unlock this feature.',
].join('\n'),
);
}
return null;
}),
pageSize: chainPropTypes(PropTypes.number, (props: any) => {
if (props.pageSize && props.pageSize > MAX_PAGE_SIZE) {
return new Error(
Expand Down
1 change: 1 addition & 0 deletions packages/grid/data-grid/src/DataGridProps.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ export type DataGridProps = Omit<
| 'licenseStatus'
| 'options'
| 'onRowsScrollEnd'
| 'onViewportRowsChange'
| 'pagination'
| 'scrollEndThreshold'
| 'signature'
Expand Down
1 change: 1 addition & 0 deletions packages/grid/data-grid/src/useDataGridProps.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ const DATA_GRID_FORCED_PROPS: Omit<
disableMultipleSelection: true,
pagination: true,
onRowsScrollEnd: undefined,
onViewportRowsChange: undefined,
checkboxSelectionVisibleOnly: false,
signature: 'DataGrid',
};
Expand Down
57 changes: 51 additions & 6 deletions packages/grid/x-grid/src/tests/events.XGrid.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,8 @@ import {
fireEvent,
// @ts-ignore
screen,
// @ts-expect-error need to migrate helpers to TypeScript
waitFor,
} from 'test/utils';
import { expect } from 'chai';
import {
Expand All @@ -21,6 +23,7 @@ import {
} from '@material-ui/x-grid';
import { getCell, getColumnHeaderCell, getRow } from 'test/utils/helperFn';
import { spy } from 'sinon';
import { useData } from 'packages/storybook/src/hooks/useData';

describe('<XGrid /> - Events Params', () => {
// TODO v5: replace with createClientRender
Expand Down Expand Up @@ -73,6 +76,11 @@ describe('<XGrid /> - Events Params', () => {
);
};

const TestVirtualization = (props: Partial<XGridProps>) => {
const data = useData(200, 10);
return <XGrid rows={data.rows} columns={data.columns} {...props} />;
};

describe('columnHeaderParams', () => {
it('should include the correct params', () => {
let eventArgs: { params: GridColumnHeaderParams; event: React.MouseEvent } | null = null;
Expand Down Expand Up @@ -261,16 +269,17 @@ describe('<XGrid /> - Events Params', () => {
expect(handleSelection.callCount).to.equal(0);
});
});

it('publishing GRID_ROWS_SCROLL should call onRowsScrollEnd callback', () => {
const handleOnRowsScrollEnd = spy();
const handleRowsScrollEnd = spy();

render(<TestEvents onRowsScrollEnd={handleOnRowsScrollEnd} />);
render(<TestEvents onRowsScrollEnd={handleRowsScrollEnd} />);
apiRef.current.publishEvent(GRID_ROWS_SCROLL);
expect(handleOnRowsScrollEnd.callCount).to.equal(1);
expect(handleRowsScrollEnd.callCount).to.equal(1);
});

it('call onRowsScrollEnd when viewport scroll reaches the bottom', () => {
const handleOnRowsScrollEnd = spy();
const handleRowsScrollEnd = spy();
const data = {
rows: [
{
Expand Down Expand Up @@ -303,13 +312,49 @@ describe('<XGrid /> - Events Params', () => {

const { container } = render(
<div style={{ width: 300, height: 300 }}>
<XGrid columns={data.columns} rows={data.rows} onRowsScrollEnd={handleOnRowsScrollEnd} />
<XGrid columns={data.columns} rows={data.rows} onRowsScrollEnd={handleRowsScrollEnd} />
</div>,
);
const gridWindow = container.querySelector('.MuiDataGrid-window');
// arbitrary number to make sure that the bottom of the grid window is reached.
gridWindow.scrollTop = 12345;
gridWindow.dispatchEvent(new Event('scroll'));
expect(handleOnRowsScrollEnd.callCount).to.equal(1);
expect(handleRowsScrollEnd.callCount).to.equal(1);
});

it('call onViewportRowsChange when the viewport rows change', async () => {
const headerHeight = 40;
const rowHeight = 40;
const pageSize = 4;
const handleViewportRowsChange = spy();
const { container } = render(
<div style={{ width: 300, height: headerHeight + rowHeight * pageSize }}>
<TestVirtualization rowHeight={rowHeight} onViewportRowsChange={handleViewportRowsChange} />
DanailH marked this conversation as resolved.
Show resolved Hide resolved
</div>,
);

await waitFor(() => {
expect(handleViewportRowsChange.callCount).to.equal(1);
expect(handleViewportRowsChange.lastCall.args[0].firstRowIndex).to.equal(0);
expect(handleViewportRowsChange.lastCall.args[0].lastRowIndex).to.equal(4); // should be pageSize + 1
});
const gridWindow = container.querySelector('.MuiDataGrid-window');
// scroll 4 rows so that the renderContext is updated. To be changed to a scroll of 1 row.
gridWindow.scrollTop = rowHeight * pageSize;
gridWindow.dispatchEvent(new Event('scroll'));
await waitFor(() => {
expect(handleViewportRowsChange.callCount).to.equal(2);
expect(handleViewportRowsChange.lastCall.args[0].firstRowIndex).to.equal(4); // should be 1
expect(handleViewportRowsChange.lastCall.args[0].lastRowIndex).to.equal(8); // should be pageSize + 1
});

// scroll another 4 rows so that the renderContext is updated. To be changed to a scroll of 2 row.
gridWindow.scrollTop = rowHeight * pageSize * 2;
gridWindow.dispatchEvent(new Event('scroll'));
await waitFor(() => {
expect(handleViewportRowsChange.callCount).to.equal(3);
expect(handleViewportRowsChange.lastCall.args[0].firstRowIndex).to.equal(8); // should be 2
expect(handleViewportRowsChange.lastCall.args[0].lastRowIndex).to.equal(12); // should be pageSize + 1
});
});
});