-
Notifications
You must be signed in to change notification settings - Fork 1.2k
/
Copy pathuseTableCell.ts
106 lines (93 loc) · 3.39 KB
/
useTableCell.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
/*
* Copyright 2020 Adobe. All rights reserved.
* This file is licensed to you under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. You may obtain a copy
* of the License at http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under
* the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
* OF ANY KIND, either express or implied. See the License for the specific language
* governing permissions and limitations under the License.
*/
import {focusSafely, getFocusableTreeWalker} from '@react-aria/focus';
import {HTMLAttributes, RefObject} from 'react';
import {isFocusVisible, usePress} from '@react-aria/interactions';
import {mergeProps} from '@react-aria/utils';
import {Node} from '@react-types/shared';
import {TableState} from '@react-stately/table';
import {useSelectableItem} from '@react-aria/selection';
interface GridCellProps {
node: Node<unknown>,
ref: RefObject<HTMLElement>,
isVirtualized?: boolean,
isDisabled?: boolean
}
interface GridCellAria {
gridCellProps: HTMLAttributes<HTMLElement>
}
export function useTableCell<T>(props: GridCellProps, state: TableState<T>): GridCellAria {
let {
node,
ref,
isVirtualized,
isDisabled
} = props;
// Handles focusing the cell. If there is a focusable child,
// it is focused, otherwise the cell itself is focused.
let focus = () => {
let treeWalker = getFocusableTreeWalker(ref.current);
let focusable = treeWalker.firstChild() as HTMLElement;
if (focusable) {
focusSafely(focusable);
state.selectionManager.setFocusWithinItem(true);
} else {
focusSafely(ref.current);
}
};
let {itemProps} = useSelectableItem({
selectionManager: state.selectionManager,
key: node.key,
ref,
isVirtualized,
focus
});
// TODO: move into useSelectableItem?
let {pressProps} = usePress({...itemProps, isDisabled});
// Grid cells can have focusable elements inside them. In this case, focus should
// be marshalled to that element rather than focusing the cell itself.
let onFocus = (e) => {
if (e.target !== ref.current) {
// useSelectableItem only handles setting the focused key when
// the focused element is the gridcell itself. We also want to
// set the focused key when a child element receives focus.
// If focus is currently visible (e.g. the user is navigating with the keyboard),
// then skip this. We want to restore focus to the previously focused row/cell
// in that case since the table should act like a single tab stop.
if (!isFocusVisible()) {
state.selectionManager.setFocusedKey(node.key);
}
return;
}
// If the cell itself is focused, wait a frame so that focus finishes propagatating
// up to the tree, and move focus to a focusable child if possible.
requestAnimationFrame(() => {
if (document.activeElement === ref.current) {
focus();
}
});
};
let onBlur = () => {
state.selectionManager.setFocusWithinItem(false);
};
let gridCellProps: HTMLAttributes<HTMLElement> = mergeProps(pressProps, {
role: 'gridcell',
onFocus,
onBlur
});
if (isVirtualized) {
gridCellProps['aria-colindex'] = node.index + 1; // aria-colindex is 1-based
}
return {
gridCellProps
};
}