Skip to content

Commit b778c8d

Browse files
authored
Column resizing (adobe#2883)
Column resizing initial work, added algorithm for calculating column widths as result of user interaction
1 parent 91ca94f commit b778c8d

24 files changed

+2586
-137
lines changed

packages/@adobe/spectrum-css-temp/components/table/index.css

+46
Original file line numberDiff line numberDiff line change
@@ -86,6 +86,44 @@ svg.spectrum-Table-sortedIcon {
8686
}
8787
}
8888

89+
.spectrum-Table--resizingColumn {
90+
.spectrum-Table-row,
91+
.spectrum-Table-headCell {
92+
cursor: col-resize;
93+
}
94+
}
95+
96+
.spectrum-Table-columnResizer {
97+
display: flex;
98+
justify-content: end;
99+
box-sizing: border-box;
100+
position: absolute;
101+
inset-block-start: 0px;
102+
inset-inline-end: 0px;
103+
inline-size: 10px;
104+
block-size: 100%;
105+
cursor: col-resize;
106+
user-select: none;
107+
108+
&::after {
109+
content: "";
110+
position: absolute;
111+
display: block;
112+
box-sizing: border-box;
113+
inline-size: 1px;
114+
block-size: 100%;
115+
background-color: var(--spectrum-table-divider-border-color);
116+
}
117+
118+
&:active,
119+
&:focus {
120+
&::after {
121+
inline-size: 2px;
122+
background-color: var(--spectrum-global-color-blue-400);
123+
}
124+
}
125+
}
126+
89127
.spectrum-Table-cell--alignCenter {
90128
text-align: center;
91129
}
@@ -236,6 +274,14 @@ svg.spectrum-Table-sortedIcon {
236274
border-inline-end-width: var(--spectrum-table-divider-border-size);
237275
}
238276

277+
.spectrum-Table-cell--divider {
278+
&.is-resizable {
279+
&:hover {
280+
border-inline-end-width: 3px;
281+
}
282+
}
283+
}
284+
239285
.spectrum-Table-row {
240286
position: relative;
241287
cursor: default;

packages/@adobe/spectrum-css-temp/components/table/skin.css

+6
Original file line numberDiff line numberDiff line change
@@ -49,6 +49,12 @@ governing permissions and limitations under the License.
4949
}
5050
}
5151
}
52+
53+
&.is-resizable {
54+
&.is-hovered {
55+
color: var(--spectrum-table-header-text-color-hover);
56+
}
57+
}
5258
}
5359

5460
/* Helper for shared drop target overlay */

packages/@react-aria/table/src/useTableColumnHeader.ts

+3-1
Original file line numberDiff line numberDiff line change
@@ -42,12 +42,14 @@ interface ColumnHeaderAria {
4242
*/
4343
export function useTableColumnHeader<T>(props: ColumnHeaderProps, state: TableState<T>, ref: RefObject<HTMLElement>): ColumnHeaderAria {
4444
let {node} = props;
45+
let allowsResizing = node.props.allowsResizing;
4546
let allowsSorting = node.props.allowsSorting;
4647
let {gridCellProps} = useGridCell(props, state, ref);
4748

4849
let isSelectionCellDisabled = node.props.isSelectionCell && state.selectionManager.selectionMode === 'single';
4950
let {pressProps} = usePress({
50-
isDisabled: !allowsSorting || isSelectionCellDisabled,
51+
// Disabled for allowsResizing because if resizing is allowed, a menu trigger is added to the column header.
52+
isDisabled: !allowsSorting || isSelectionCellDisabled || allowsResizing,
5153
onPress() {
5254
state.sort(node.key);
5355
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
/*
2+
* Copyright 2020 Adobe. All rights reserved.
3+
* This file is licensed to you under the Apache License, Version 2.0 (the "License");
4+
* you may not use this file except in compliance with the License. You may obtain a copy
5+
* of the License at http://www.apache.org/licenses/LICENSE-2.0
6+
*
7+
* Unless required by applicable law or agreed to in writing, software distributed under
8+
* the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
9+
* OF ANY KIND, either express or implied. See the License for the specific language
10+
* governing permissions and limitations under the License.
11+
*/
12+
13+
import {focusSafely, useFocusable} from '@react-aria/focus';
14+
import {mergeProps} from '@react-aria/utils';
15+
import {useKeyboard, useMove} from '@react-aria/interactions';
16+
import {useLocale} from '@react-aria/i18n';
17+
import {useRef} from 'react';
18+
19+
export function useTableColumnResize(state, item, ref): any {
20+
const stateRef = useRef(null);
21+
// keep track of what the cursor on the body is so it can be restored back to that when done resizing
22+
const cursor = useRef(null);
23+
stateRef.current = state;
24+
25+
let {direction} = useLocale();
26+
let {focusableProps} = useFocusable({excludeFromTabOrder: true}, ref);
27+
let {keyboardProps} = useKeyboard({
28+
onKeyDown: (e) => {
29+
if (e.key === 'Tab') {
30+
// useKeyboard stops propagation by default. We want to continue propagation for tab so focus leaves the table
31+
e.continuePropagation();
32+
}
33+
if (e.key === 'Escape' || e.key === 'Enter' || e.key === ' ') {
34+
// switch focus back to the column header on escape
35+
const columnHeader = ref.current.previousSibling;
36+
if (columnHeader) {
37+
focusSafely(columnHeader);
38+
}
39+
}
40+
}
41+
});
42+
43+
const columnResizeWidthRef = useRef(null);
44+
const {moveProps} = useMove({
45+
onMoveStart() {
46+
stateRef.current.onColumnResizeStart();
47+
columnResizeWidthRef.current = stateRef.current.getColumnWidth(item.key);
48+
cursor.current = document.body.style.cursor;
49+
document.body.style.setProperty('cursor', 'col-resize');
50+
},
51+
onMove({deltaX, pointerType}) {
52+
if (direction === 'rtl') {
53+
deltaX *= -1;
54+
}
55+
// if moving up/down only, no need to resize
56+
if (deltaX !== 0) {
57+
if (pointerType === 'keyboard') {
58+
deltaX *= 10;
59+
}
60+
columnResizeWidthRef.current += deltaX;
61+
stateRef.current.onColumnResize(item, columnResizeWidthRef.current);
62+
}
63+
},
64+
onMoveEnd() {
65+
stateRef.current.onColumnResizeEnd();
66+
columnResizeWidthRef.current = 0;
67+
document.body.style.cursor = cursor.current;
68+
}
69+
});
70+
71+
return {
72+
resizerProps: {
73+
...mergeProps(moveProps, focusableProps, keyboardProps)
74+
}
75+
};
76+
}

packages/@react-spectrum/table/package.json

+3
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@
3131
"url": "https://github.com/adobe/react-spectrum"
3232
},
3333
"dependencies": {
34+
"@react-aria/button": "^3.4.1",
3435
"@babel/runtime": "^7.6.2",
3536
"@react-aria/focus": "^3.5.3",
3637
"@react-aria/grid": "^3.2.4",
@@ -40,7 +41,9 @@
4041
"@react-aria/utils": "^3.11.3",
4142
"@react-aria/virtualizer": "^3.3.8",
4243
"@react-aria/visually-hidden": "^3.2.6",
44+
"@react-spectrum/button": "^3.7.1",
4345
"@react-spectrum/checkbox": "^3.3.2",
46+
"@react-spectrum/menu": "^3.6.1",
4447
"@react-spectrum/progress": "^3.1.6",
4548
"@react-spectrum/tooltip": "^3.1.7",
4649
"@react-spectrum/utils": "^3.6.6",
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
/* eslint-disable jsx-a11y/role-supports-aria-props */
2+
import {classNames} from '@react-spectrum/utils';
3+
import {FocusRing} from '@react-aria/focus';
4+
import React from 'react';
5+
import styles from '@adobe/spectrum-css-temp/components/table/vars.css';
6+
import {useTableColumnResize} from '@react-aria/table/src/useTableColumnResize';
7+
import {useTableContext} from './TableView';
8+
9+
10+
function Resizer(props, ref) {
11+
const {item} = props;
12+
let state = useTableContext();
13+
let {resizerProps} = useTableColumnResize(state, item, ref);
14+
return (
15+
<FocusRing focusRingClass={classNames(styles, 'focus-ring')}>
16+
<div
17+
ref={ref}
18+
{...resizerProps}
19+
className={classNames(styles, 'spectrum-Table-columnResizer')}
20+
role="separator"
21+
aria-orientation="vertical"
22+
aria-label="Resize column"
23+
aria-labelledby={item.key}
24+
aria-valuenow={state.getColumnWidth(item.key)}
25+
aria-valuemin={state.getColumnMinWidth(item.key)}
26+
aria-valuemax={state.getColumnMaxWidth(item.key)} />
27+
</FocusRing>
28+
);
29+
}
30+
31+
const _Resizer = React.forwardRef(Resizer);
32+
export {_Resizer as Resizer};

0 commit comments

Comments
 (0)