Skip to content

Commit ab5c648

Browse files
shleewhitedchyun
andauthored
AdvancedTable: follow ups (#2659)
Co-authored-by: Dylan Hyun <dylan.hyun@hashicorp.com>
1 parent 31497e7 commit ab5c648

File tree

18 files changed

+583
-568
lines changed

18 files changed

+583
-568
lines changed

.changeset/thin-masks-attack.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"@hashicorp/design-system-components": patch
3+
---
4+
5+
`AdvancedTable` - Refactored keyboard navigation to a new modifier `hds-advanced-table-cell` for reusability, and disabled default behavior for arrow keys in focused cells.

packages/components/package.json

+3-1
Original file line numberDiff line numberDiff line change
@@ -135,7 +135,6 @@
135135
"./components/hds/accordion/item/button.js": "./dist/_app_/components/hds/accordion/item/button.js",
136136
"./components/hds/accordion/item/index.js": "./dist/_app_/components/hds/accordion/item/index.js",
137137
"./components/hds/advanced-table/expandable-tr-group.js": "./dist/_app_/components/hds/advanced-table/expandable-tr-group.js",
138-
"./components/hds/advanced-table/helpers.js": "./dist/_app_/components/hds/advanced-table/helpers.js",
139138
"./components/hds/advanced-table/index.js": "./dist/_app_/components/hds/advanced-table/index.js",
140139
"./components/hds/advanced-table/td.js": "./dist/_app_/components/hds/advanced-table/td.js",
141140
"./components/hds/advanced-table/th-button-expand.js": "./dist/_app_/components/hds/advanced-table/th-button-expand.js",
@@ -325,6 +324,9 @@
325324
"./helpers/hds-link-to-models.js": "./dist/_app_/helpers/hds-link-to-models.js",
326325
"./helpers/hds-link-to-query.js": "./dist/_app_/helpers/hds-link-to-query.js",
327326
"./instance-initializers/load-sprite.js": "./dist/_app_/instance-initializers/load-sprite.js",
327+
"./modifiers/hds-advanced-table-cell.js": "./dist/_app_/modifiers/hds-advanced-table-cell.js",
328+
"./modifiers/hds-advanced-table-cell/dom-management.js": "./dist/_app_/modifiers/hds-advanced-table-cell/dom-management.js",
329+
"./modifiers/hds-advanced-table-cell/keyboard-navigation.js": "./dist/_app_/modifiers/hds-advanced-table-cell/keyboard-navigation.js",
328330
"./modifiers/hds-anchored-position.js": "./dist/_app_/modifiers/hds-anchored-position.js",
329331
"./modifiers/hds-clipboard.js": "./dist/_app_/modifiers/hds-clipboard.js",
330332
"./modifiers/hds-code-editor.js": "./dist/_app_/modifiers/hds-code-editor.js",

packages/components/src/components/hds/advanced-table/index.hbs

+1-2
Original file line numberDiff line numberDiff line change
@@ -9,8 +9,7 @@
99
role="grid"
1010
aria-describedby={{this._captionId}}
1111
{{style grid-template-columns=this.gridTemplateColumns}}
12-
{{did-insert this.didInsert}}
13-
{{will-destroy this.willDestroy}}
12+
{{this._setUpObserver}}
1413
>
1514
<div id={{this._captionId}} class="sr-only hds-advanced-table__caption" aria-live="polite">
1615
{{@caption}}

packages/components/src/components/hds/advanced-table/index.ts

+9-10
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ import { assert } from '@ember/debug';
99
import { tracked } from '@glimmer/tracking';
1010
import type { ComponentLike } from '@glint/template';
1111
import { guidFor } from '@ember/object/internals';
12-
import config from 'ember-get-config';
12+
import { modifier } from 'ember-modifier';
1313

1414
import {
1515
HdsAdvancedTableDensityValues,
@@ -282,12 +282,12 @@ export default class HdsAdvancedTable extends Component<HdsAdvancedTableSignatur
282282
return classes.join(' ');
283283
}
284284

285-
@action didInsert(element: HTMLDivElement): void {
285+
private _setUpObserver = modifier((element: HTMLElement) => {
286286
const stickyGridHeader = element.querySelector(
287287
'.hds-advanced-table__thead.hds-advanced-table__thead--sticky'
288288
);
289289

290-
if (stickyGridHeader !== null && config.environment !== 'test') {
290+
if (stickyGridHeader !== null) {
291291
this._observer = new IntersectionObserver(
292292
([element]) =>
293293
element?.target.classList.toggle(
@@ -299,14 +299,13 @@ export default class HdsAdvancedTable extends Component<HdsAdvancedTableSignatur
299299

300300
this._observer.observe(stickyGridHeader);
301301
}
302-
}
303302

304-
@action willDestroy() {
305-
super.willDestroy();
306-
if (this._observer) {
307-
this._observer.disconnect();
308-
}
309-
}
303+
return () => {
304+
if (this._observer) {
305+
this._observer.disconnect();
306+
}
307+
};
308+
});
310309

311310
@action
312311
setSortBy(column: string): void {

packages/components/src/components/hds/advanced-table/td.hbs

+5-3
Original file line numberDiff line numberDiff line change
@@ -8,12 +8,14 @@
88
aria-rowspan={{@rowspan}}
99
aria-colspan={{@colspan}}
1010
{{style grid-row=this.rowspan grid-column=this.colspan}}
11-
{{did-insert this.didInsert}}
12-
{{will-destroy this.willDestroy}}
11+
{{hds-advanced-table-cell
12+
handleEnableFocusTrap=this.enableFocusTrap
13+
shouldTrapFocus=this._shouldTrapFocus
14+
setCellElement=this.setElement
15+
}}
1316
{{! @glint-expect-error - https://github.com/josemarluedke/ember-focus-trap/issues/86 }}
1417
{{focus-trap
1518
isActive=this._shouldTrapFocus
16-
shouldSelfFocus=true
1719
focusTrapOptions=(hash
1820
onDeactivate=this.onFocusTrapDeactivate initialFocus=this.getInitialFocus clickOutsideDeactivates=true
1921
)

packages/components/src/components/hds/advanced-table/td.ts

+3-34
Original file line numberDiff line numberDiff line change
@@ -11,12 +11,7 @@ import { focusable, type FocusableElement } from 'tabbable';
1111

1212
import type { HdsAdvancedTableHorizontalAlignment } from './types.ts';
1313
import { HdsAdvancedTableHorizontalAlignmentValues } from './types.ts';
14-
import {
15-
didInsertGridCell,
16-
handleGridCellKeyPress,
17-
updateTabbableChildren,
18-
onFocusTrapDeactivate,
19-
} from './helpers.ts';
14+
import { onFocusTrapDeactivate } from '../../../modifiers/hds-advanced-table-cell/dom-management.ts';
2015

2116
export const ALIGNMENTS: string[] = Object.values(
2217
HdsAdvancedTableHorizontalAlignmentValues
@@ -37,7 +32,6 @@ export interface HdsAdvancedTableTdSignature {
3732
export default class HdsAdvancedTableTd extends Component<HdsAdvancedTableTdSignature> {
3833
@tracked private _shouldTrapFocus = false;
3934
private _element!: HTMLDivElement;
40-
private _observer: MutationObserver | undefined = undefined;
4135

4236
// rowspan and colspan have to return 'auto' if not defined because otherwise the style modifier sets grid-area: undefined on the cell, which breaks the grid styles
4337
get rowspan(): string {
@@ -67,11 +61,7 @@ export default class HdsAdvancedTableTd extends Component<HdsAdvancedTableTdSign
6761
}
6862

6963
get classNames(): string {
70-
const classes = [
71-
'hds-advanced-table__td',
72-
'hds-typography-body-200',
73-
'hds-font-weight-regular',
74-
];
64+
const classes = ['hds-advanced-table__td'];
7565

7666
// add a class based on the @align argument
7767
if (this.align) {
@@ -95,28 +85,7 @@ export default class HdsAdvancedTableTd extends Component<HdsAdvancedTableTdSign
9585
return cellFocusableElements[0];
9686
}
9787

98-
@action
99-
didInsert(element: HTMLDivElement): void {
88+
@action setElement(element: HTMLDivElement): void {
10089
this._element = element;
101-
didInsertGridCell(element);
102-
element.addEventListener('keydown', (event: KeyboardEvent) => {
103-
handleGridCellKeyPress(event, this.enableFocusTrap);
104-
});
105-
106-
this._observer = new MutationObserver(() => {
107-
updateTabbableChildren(this._element, this._shouldTrapFocus);
108-
});
109-
110-
this._observer.observe(this._element, {
111-
childList: true,
112-
subtree: true,
113-
});
114-
}
115-
116-
@action willDestroy() {
117-
super.willDestroy();
118-
if (this._observer) {
119-
this._observer.disconnect();
120-
}
12190
}
12291
}

packages/components/src/components/hds/advanced-table/th-sort.hbs

+5-3
Original file line numberDiff line numberDiff line change
@@ -8,12 +8,14 @@
88
role="columnheader"
99
aria-rowspan={{@rowspan}}
1010
aria-colspan={{@colspan}}
11-
{{did-insert this.didInsert}}
12-
{{will-destroy this.willDestroy}}
11+
{{hds-advanced-table-cell
12+
handleEnableFocusTrap=this.enableFocusTrap
13+
shouldTrapFocus=this._shouldTrapFocus
14+
setCellElement=this.setElement
15+
}}
1316
{{! @glint-expect-error - https://github.com/josemarluedke/ember-focus-trap/issues/86 }}
1417
{{focus-trap
1518
isActive=this._shouldTrapFocus
16-
shouldSelfFocus=true
1719
focusTrapOptions=(hash
1820
onDeactivate=this.onFocusTrapDeactivate initialFocus=this.getInitialFocus clickOutsideDeactivates=true
1921
)

packages/components/src/components/hds/advanced-table/th-sort.ts

+2-29
Original file line numberDiff line numberDiff line change
@@ -21,12 +21,7 @@ import type {
2121
HdsAdvancedTableThSortOrderLabels,
2222
} from './types.ts';
2323
import type { HdsAdvancedTableThButtonSortSignature } from './th-button-sort.ts';
24-
import {
25-
didInsertGridCell,
26-
handleGridCellKeyPress,
27-
onFocusTrapDeactivate,
28-
updateTabbableChildren,
29-
} from './helpers.ts';
24+
import { onFocusTrapDeactivate } from '../../../modifiers/hds-advanced-table-cell/dom-management.ts';
3025

3126
export const ALIGNMENTS: string[] = Object.values(
3227
HdsAdvancedTableHorizontalAlignmentValues
@@ -52,7 +47,6 @@ export default class HdsAdvancedTableThSort extends Component<HdsAdvancedTableTh
5247
private _labelId = guidFor(this);
5348
private _element!: HTMLDivElement;
5449
@tracked private _shouldTrapFocus = false;
55-
private _observer: MutationObserver | undefined = undefined;
5650

5751
get ariaSort(): HdsAdvancedTableThSortOrderLabels {
5852
switch (this.args.sortOrder) {
@@ -103,28 +97,7 @@ export default class HdsAdvancedTableThSort extends Component<HdsAdvancedTableTh
10397
return cellFocusableElements[0];
10498
}
10599

106-
@action
107-
didInsert(element: HTMLDivElement): void {
100+
@action setElement(element: HTMLDivElement): void {
108101
this._element = element;
109-
didInsertGridCell(element);
110-
element.addEventListener('keydown', (event: KeyboardEvent) => {
111-
handleGridCellKeyPress(event, this.enableFocusTrap);
112-
});
113-
114-
this._observer = new MutationObserver(() => {
115-
updateTabbableChildren(this._element, this._shouldTrapFocus);
116-
});
117-
118-
this._observer.observe(this._element, {
119-
childList: true,
120-
subtree: true,
121-
});
122-
}
123-
124-
@action willDestroy() {
125-
super.willDestroy();
126-
if (this._observer) {
127-
this._observer.disconnect();
128-
}
129102
}
130103
}

packages/components/src/components/hds/advanced-table/th.hbs

+5-3
Original file line numberDiff line numberDiff line change
@@ -9,12 +9,14 @@
99
aria-colspan={{@colspan}}
1010
aria-describedby={{@parentId}}
1111
{{style grid-row=this.rowspan grid-column=this.colspan padding-left=this.paddingLeft}}
12-
{{did-insert this.didInsert}}
13-
{{will-destroy this.willDestroy}}
12+
{{hds-advanced-table-cell
13+
handleEnableFocusTrap=this.enableFocusTrap
14+
shouldTrapFocus=this._shouldTrapFocus
15+
setCellElement=this.setElement
16+
}}
1417
{{! @glint-expect-error - https://github.com/josemarluedke/ember-focus-trap/issues/86 }}
1518
{{focus-trap
1619
isActive=this._shouldTrapFocus
17-
shouldSelfFocus=true
1820
focusTrapOptions=(hash
1921
onDeactivate=this.onFocusTrapDeactivate initialFocus=this.getInitialFocus clickOutsideDeactivates=true
2022
)

packages/components/src/components/hds/advanced-table/th.ts

+2-29
Original file line numberDiff line numberDiff line change
@@ -15,12 +15,7 @@ import type {
1515
HdsAdvancedTableScope,
1616
} from './types.ts';
1717
import { HdsAdvancedTableHorizontalAlignmentValues } from './types.ts';
18-
import {
19-
didInsertGridCell,
20-
handleGridCellKeyPress,
21-
onFocusTrapDeactivate,
22-
updateTabbableChildren,
23-
} from './helpers.ts';
18+
import { onFocusTrapDeactivate } from '../../../modifiers/hds-advanced-table-cell/dom-management.ts';
2419

2520
export const ALIGNMENTS: string[] = Object.values(
2621
HdsAdvancedTableHorizontalAlignmentValues
@@ -52,7 +47,6 @@ export default class HdsAdvancedTableTh extends Component<HdsAdvancedTableThSign
5247
private _labelId = this.args.newLabel ? this.args.newLabel : guidFor(this);
5348
private _element!: HTMLDivElement;
5449
@tracked private _shouldTrapFocus = false;
55-
private _observer: MutationObserver | undefined = undefined;
5650

5751
get scope(): HdsAdvancedTableScope {
5852
const { scope = 'col' } = this.args;
@@ -122,28 +116,7 @@ export default class HdsAdvancedTableTh extends Component<HdsAdvancedTableThSign
122116
return cellFocusableElements[0];
123117
}
124118

125-
@action
126-
didInsert(element: HTMLDivElement): void {
119+
@action setElement(element: HTMLDivElement): void {
127120
this._element = element;
128-
didInsertGridCell(element);
129-
element.addEventListener('keydown', (event: KeyboardEvent) => {
130-
handleGridCellKeyPress(event, this.enableFocusTrap);
131-
});
132-
133-
this._observer = new MutationObserver(() => {
134-
updateTabbableChildren(this._element, this._shouldTrapFocus);
135-
});
136-
137-
this._observer.observe(this._element, {
138-
childList: true,
139-
subtree: true,
140-
});
141-
}
142-
143-
@action willDestroy() {
144-
super.willDestroy();
145-
if (this._observer) {
146-
this._observer.disconnect();
147-
}
148121
}
149122
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,83 @@
1+
/**
2+
* Copyright (c) HashiCorp, Inc.
3+
* SPDX-License-Identifier: MPL-2.0
4+
*/
5+
6+
import Modifier from 'ember-modifier';
7+
import type { ArgsFor, PositionalArgs } from 'ember-modifier';
8+
import { registerDestructor } from '@ember/destroyable';
9+
10+
import {
11+
didInsertGridCell,
12+
updateTabbableChildren,
13+
} from './hds-advanced-table-cell/dom-management.ts';
14+
import { handleGridCellKeyPress } from './hds-advanced-table-cell/keyboard-navigation.ts';
15+
16+
export interface HdsAdvancedTableCellModifierSignature {
17+
Args: {
18+
Named: {
19+
handleEnableFocusTrap: () => void;
20+
shouldTrapFocus: boolean;
21+
setCellElement: (el: HTMLDivElement) => void;
22+
};
23+
};
24+
Element: HTMLDivElement;
25+
}
26+
27+
function cleanup(instance: HdsAdvancedTableCellModifier): void {
28+
const { _observer } = instance;
29+
if (_observer) {
30+
_observer.disconnect();
31+
}
32+
}
33+
34+
export default class HdsAdvancedTableCellModifier extends Modifier<HdsAdvancedTableCellModifierSignature> {
35+
// have a copy of shouldTrapFocus internally so the correct value is used when update tabbable children
36+
private _shouldTrapFocus = false;
37+
private _didSetup = false;
38+
_observer: MutationObserver | undefined = undefined;
39+
40+
constructor(
41+
owner: unknown,
42+
args: ArgsFor<HdsAdvancedTableCellModifierSignature>
43+
) {
44+
super(owner, args);
45+
registerDestructor(this, cleanup);
46+
}
47+
48+
modify(
49+
element: HTMLDivElement,
50+
positional: PositionalArgs<HdsAdvancedTableCellModifierSignature>,
51+
named: HdsAdvancedTableCellModifierSignature['Args']['Named']
52+
): void {
53+
this._shouldTrapFocus = named.shouldTrapFocus;
54+
55+
if (!this._didSetup) {
56+
this.#setupObserver(element, positional, named);
57+
named.setCellElement(element);
58+
this._didSetup = true;
59+
}
60+
}
61+
62+
#setupObserver(
63+
element: HdsAdvancedTableCellModifierSignature['Element'],
64+
_positional: PositionalArgs<HdsAdvancedTableCellModifierSignature>,
65+
named: HdsAdvancedTableCellModifierSignature['Args']['Named']
66+
): void {
67+
const { handleEnableFocusTrap } = named;
68+
69+
didInsertGridCell(element);
70+
element.addEventListener('keydown', (event: KeyboardEvent) => {
71+
handleGridCellKeyPress(event, handleEnableFocusTrap);
72+
});
73+
74+
this._observer = new MutationObserver(() => {
75+
updateTabbableChildren(element, this._shouldTrapFocus);
76+
});
77+
78+
this._observer.observe(element, {
79+
childList: true,
80+
subtree: true,
81+
});
82+
}
83+
}

0 commit comments

Comments
 (0)