Skip to content

Commit 1ec4cef

Browse files
authored
fix: Fixes table sticky cells clicks (#3354)
1 parent 407c869 commit 1ec4cef

5 files changed

+225
-10
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
2+
// SPDX-License-Identifier: Apache-2.0
3+
4+
import { BasePageObject } from '@cloudscape-design/browser-test-tools/page-objects';
5+
import useBrowser from '@cloudscape-design/browser-test-tools/use-browser';
6+
7+
import createWrapper from '../../../lib/components/test-utils/selectors';
8+
9+
import styles from '../../../lib/components/table/styles.selectors.js';
10+
11+
test(
12+
'does not cause table wrapper to scroll when clicking on an interactive element inside a sticky column',
13+
useBrowser(async browser => {
14+
const tableWrapper = createWrapper().findTable('[data-test-id="small-table"]');
15+
const wrapperSelector = tableWrapper.findAllByClassName(styles.wrapper).toSelector();
16+
await browser.setWindowSize(600, 1000);
17+
await browser.url('#/light/table/sticky-columns/?selectionType=multi&stickyColumnsFirst=1&stickyColumnsLast=1');
18+
const page = new BasePageObject(browser);
19+
await page.waitForVisible(tableWrapper.findRows().toSelector());
20+
21+
await page.elementScrollTo(wrapperSelector, { top: 0, left: 100 });
22+
await expect(page.getElementScroll(wrapperSelector)).resolves.toEqual({ top: 0, left: 100 });
23+
24+
await page.click(tableWrapper.findRowSelectionArea(1).toSelector());
25+
await page.pause(100);
26+
await expect(page.getElementScroll(wrapperSelector)).resolves.toEqual({ top: 0, left: 100 });
27+
})
28+
);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,92 @@
1+
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
2+
// SPDX-License-Identifier: Apache-2.0
3+
4+
import * as React from 'react';
5+
import { render } from '@testing-library/react';
6+
7+
import { usePreventStickyClickScroll } from '../../../lib/components/table/use-prevent-sticky-click-scroll';
8+
9+
import bodyCellStyles from '../../../lib/components/table/body-cell/styles.css.js';
10+
11+
const TestComponent = () => {
12+
const wrapperRef = React.useRef<HTMLDivElement>(null);
13+
14+
usePreventStickyClickScroll(wrapperRef);
15+
16+
return (
17+
<div ref={wrapperRef} data-testid="wrapper">
18+
<table data-testid="table-root">
19+
<tbody>
20+
<tr>
21+
<td className={bodyCellStyles['sticky-cell']}>
22+
<button data-testid="r0c0" type="button">
23+
r0c0
24+
</button>
25+
</td>
26+
<td>
27+
<button data-testid="r0c1" type="button">
28+
r0c1
29+
</button>
30+
</td>
31+
<td>
32+
<button data-testid="r0c2" type="button">
33+
r0c2
34+
</button>
35+
</td>
36+
</tr>
37+
</tbody>
38+
</table>
39+
</div>
40+
);
41+
};
42+
43+
describe('usePreventStickyClickScroll', () => {
44+
beforeEach(() => {
45+
jest.clearAllMocks();
46+
});
47+
48+
it('does not reset scroll when clicking inside non-sticky cell', () => {
49+
const { getByTestId } = render(<TestComponent />);
50+
const wrapper = getByTestId('wrapper');
51+
const cell1 = getByTestId('r0c1');
52+
53+
// Imitate user scroll before clicking on a cell
54+
wrapper.scrollLeft = 20;
55+
wrapper.dispatchEvent(new Event('scroll'));
56+
57+
// Imitate user click on a non-sticky cell
58+
cell1.click();
59+
60+
// Imitate automatic browser scroll
61+
wrapper.scrollLeft = 0;
62+
wrapper.dispatchEvent(new Event('scroll'));
63+
64+
expect(wrapper.scrollLeft).toBe(0);
65+
});
66+
67+
it('does reset scroll when clicking inside sticky cell', async () => {
68+
const { getByTestId } = render(<TestComponent />);
69+
const wrapper = getByTestId('wrapper');
70+
const cell0 = getByTestId('r0c0');
71+
72+
// Imitate user scroll before clicking on a cell
73+
wrapper.scrollLeft = 20;
74+
wrapper.dispatchEvent(new Event('scroll'));
75+
76+
// Imitate user click on a sticky cell
77+
cell0.click();
78+
79+
// Imitate automatic browser scroll
80+
wrapper.scrollLeft = 0;
81+
wrapper.dispatchEvent(new Event('scroll'));
82+
83+
expect(wrapper.scrollLeft).toBe(20);
84+
85+
// Imitate user scroll after timeout has passed
86+
await new Promise(resolve => setTimeout(resolve, 50));
87+
wrapper.scrollLeft = 0;
88+
wrapper.dispatchEvent(new Event('scroll'));
89+
90+
expect(wrapper.scrollLeft).toBe(0);
91+
});
92+
});

src/table/body-cell/styles.scss

+13-9
Original file line numberDiff line numberDiff line change
@@ -31,12 +31,22 @@ $cell-offset: calc(#{awsui.$space-m} + #{awsui.$space-xs});
3131
// Ensuring enough space for absolute-positioned focus outlines of focus-able cell content elements.
3232
$cell-negative-space-vertical: 2px;
3333

34+
@mixin safe-focus-highlight($params) {
35+
@include styles.focus-highlight($params);
36+
37+
// @mixin focus-highlight sets cell's position to "relative".
38+
// Reinforcing sticky position for it to take precedence.
39+
&.sticky-cell {
40+
position: sticky;
41+
}
42+
}
43+
3444
@mixin cell-focus-outline {
35-
@include styles.focus-highlight(calc(-1 * #{awsui.$space-scaled-xxs}));
45+
@include safe-focus-highlight(calc(-1 * #{awsui.$space-scaled-xxs}));
3646

3747
// Give extra space on the left (inline start) to compensate for missing body cell padding.
3848
&.is-visual-refresh:first-child {
39-
@include styles.focus-highlight(
49+
@include safe-focus-highlight(
4050
(
4151
'vertical': calc(-1 * #{awsui.$space-scaled-xxs}),
4252
'horizontal': (
@@ -46,12 +56,6 @@ $cell-negative-space-vertical: 2px;
4656
)
4757
);
4858
}
49-
50-
// @mixin focus-highlight sets cell's position to "relative".
51-
// Reinforcing sticky position for it to take precedence.
52-
&.sticky-cell {
53-
position: sticky;
54-
}
5559
}
5660

5761
.expandable-toggle-wrapper {
@@ -396,7 +400,7 @@ $cell-negative-space-vertical: 2px;
396400
&-focusable {
397401
@include focus-visible.when-visible {
398402
// Making focus outline slightly smaller to not intersect with the success indicator.
399-
@include styles.focus-highlight(-1px);
403+
@include safe-focus-highlight(-1px);
400404
}
401405
}
402406
}

src/table/internal.tsx

+4-1
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,7 @@ import Thead, { TheadProps } from './thead';
5656
import ToolsHeader from './tools-header';
5757
import { useCellEditing } from './use-cell-editing';
5858
import { ColumnWidthDefinition, ColumnWidthsProvider, DEFAULT_COLUMN_WIDTH } from './use-column-widths';
59+
import { usePreventStickyClickScroll } from './use-prevent-sticky-click-scroll';
5960
import { useRowEvents } from './use-row-events';
6061
import useTableFocusNavigation from './use-table-focus-navigation';
6162
import { checkSortingState, getColumnKey, getItemKey, getVisibleColumnDefinitions, toContainerVariant } from './utils';
@@ -266,7 +267,7 @@ const InternalTable = React.forwardRef(
266267
[cancelEdit]
267268
);
268269

269-
const wrapperRefObject = useRef(null);
270+
const wrapperRefObject = useRef<HTMLDivElement>(null);
270271
const handleScroll = useScrollSync([wrapperRefObject, scrollbarRef, secondaryWrapperRef]);
271272

272273
const { moveFocusDown, moveFocusUp, moveFocus } = useSelectionFocusMove(selectionType, allItems.length);
@@ -379,6 +380,8 @@ const InternalTable = React.forwardRef(
379380
setLastUserAction,
380381
};
381382

383+
usePreventStickyClickScroll(wrapperRefObject);
384+
382385
const wrapperRef = useMergeRefs(wrapperRefObject, stickyState.refs.wrapper);
383386
const tableRef = useMergeRefs(tableMeasureRef, tableRefObject, stickyState.refs.table);
384387

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,88 @@
1+
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
2+
// SPDX-License-Identifier: Apache-2.0
3+
4+
import { useEffect } from 'react';
5+
6+
import bodyCellStyles from './body-cell/styles.css.js';
7+
8+
const stickyCellSelector = `.${bodyCellStyles['sticky-cell']}`;
9+
10+
// The scroll lock timeout value was determined empirically.
11+
// It is small enough to not interfere with sequential user events, and long enough
12+
// to ensure the browser-triggered scroll event occurs.
13+
const scrollLockTimeout = 50;
14+
15+
// The function provides a workaround for a Chrome issue causing unexpected scroll when clicking on interactive elements
16+
// inside sticky table cells.
17+
// When an interactive element (cell editor button, row selector, or a custom interactive button or link) is clicked, it receives
18+
// focus. The browser then tries to ensure the focused element is visible on screen, and it scrolls the element into view as needed.
19+
// In Chrome, this scrolling also occurs when clicking an interactive element inside a sticky cell, despite it being fully visible.
20+
// This causes an unneeded and unexpected scroll of the table wrapper towards the sticky element (on the left or on the right).
21+
//
22+
// Note: If moving focus to an interactive element using the keyboard, the automatic scroll still happens.
23+
// That is because the implemented workaround is not suitable for focusin events due to a difference in events order.
24+
export function usePreventStickyClickScroll(wrapperRefObject: React.RefObject<HTMLDivElement>) {
25+
useEffect(() => {
26+
if (wrapperRefObject.current) {
27+
const wrapperEl = wrapperRefObject.current;
28+
const scrollLock = new ScrollLock();
29+
30+
// For click events inside sticky cells we capture the table wrapper scroll offset.
31+
// This is used to reset the browser-enforced scrolling that is to follow.
32+
// The scroll lock is automatically cleared after a short delay.
33+
const onClick = (event: Event) => {
34+
if (
35+
event.target &&
36+
event.target instanceof HTMLElement &&
37+
(event.target.matches(stickyCellSelector) || event.target.closest(stickyCellSelector))
38+
) {
39+
scrollLock.set(wrapperEl.scrollLeft);
40+
}
41+
};
42+
wrapperEl.addEventListener('click', onClick);
43+
44+
// We cannot prevent the browser-issued scroll events from happening, and cannot cancel the default behavior.
45+
// Instead, if we detect a scroll event that immediately follows a click inside a sticky cell, we negate the
46+
// effect of it by resetting the wrapper scroll offset to its previous value.
47+
const onScroll = () => {
48+
if (scrollLock.active) {
49+
wrapperEl.scrollLeft = scrollLock.scrollLeft;
50+
scrollLock.clear();
51+
}
52+
};
53+
wrapperEl.addEventListener('scroll', onScroll);
54+
55+
return () => {
56+
wrapperEl.removeEventListener('click', onClick);
57+
wrapperEl.removeEventListener('scroll', onScroll);
58+
};
59+
}
60+
}, [wrapperRefObject]);
61+
}
62+
63+
class ScrollLock {
64+
#timeoutId = setTimeout(() => {}, 0);
65+
#scrollLeft = 0;
66+
#active = false;
67+
68+
public set(scrollLeft: number) {
69+
if (!this.#active) {
70+
this.#active = true;
71+
this.#scrollLeft = scrollLeft;
72+
this.#timeoutId = setTimeout(() => (this.#active = false), scrollLockTimeout);
73+
}
74+
}
75+
76+
public clear() {
77+
this.#active = false;
78+
clearTimeout(this.#timeoutId);
79+
}
80+
81+
public get active() {
82+
return this.#active;
83+
}
84+
85+
public get scrollLeft() {
86+
return this.#scrollLeft;
87+
}
88+
}

0 commit comments

Comments
 (0)