Skip to content

Commit

Permalink
feat(select): add keyboard support for arrow end and home
Browse files Browse the repository at this point in the history
PiperOrigin-RevId: 592643729
  • Loading branch information
Elliott Marquez authored and copybara-github committed Dec 20, 2023
1 parent 9c5cff8 commit 8912019
Show file tree
Hide file tree
Showing 2 changed files with 35 additions and 7 deletions.
4 changes: 1 addition & 3 deletions select/demo/stories.ts
Original file line number Diff line number Diff line change
Expand Up @@ -36,10 +36,8 @@ const selects: MaterialStoryInit<StoryKnobs> = {
name: 'Selects',
render(knobs) {
return html`
<div
style="display: flex; gap: 16px;flex-direction: column;align-items:end;border: 1px solid black;">
<div style="display: flex; gap: 16px;">
<md-filled-select
style="min-width: 100px;width: fit-content;"
.label=${knobs.label}
.quick=${knobs.quick}
.required=${knobs.required}
Expand Down
38 changes: 34 additions & 4 deletions select/internal/select.ts
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ import {SelectValidator} from '../../labs/behaviors/validators/select-validator.
import {getActiveItem} from '../../list/internal/list-navigation-helpers.js';
import {
CloseMenuEvent,
FocusState,
isElementInSubtree,
isSelectableKey,
} from '../../menu/internal/controllers/shared.js';
Expand Down Expand Up @@ -157,7 +158,7 @@ export abstract class Select extends selectBaseClass {
* Whether the menu should be aligned to the start or the end of the select's
* textbox.
*/
@property({attribute: 'menu-align'}) menuAlign: 'start'|'end' = 'start';
@property({attribute: 'menu-align'}) menuAlign: 'start' | 'end' = 'start';

/**
* The value of the currently selected option.
Expand Down Expand Up @@ -248,6 +249,7 @@ export abstract class Select extends selectBaseClass {

@state() private focused = false;
@state() private open = false;
@state() private defaultFocus: FocusState = FocusState.NONE;
@query('.field') private readonly field!: Field | null;
@query('md-menu') private readonly menu!: Menu | null;
@query('#label') private readonly labelEl!: HTMLElement;
Expand Down Expand Up @@ -466,7 +468,7 @@ export abstract class Select extends selectBaseClass {
return html`<div class="menu-wrapper">
<md-menu
id="listbox"
default-focus="none"
.defaultFocus=${this.defaultFocus}
role="listbox"
tabindex="-1"
aria-label=${ariaLabel || nothing}
Expand All @@ -484,8 +486,8 @@ export abstract class Select extends selectBaseClass {
.quick=${this.quick}
.positioning=${this.menuPositioning}
.typeaheadDelay=${this.typeaheadDelay}
.anchorCorner=${this.menuAlign === 'start'? 'end-start' : 'end-end'}
.menuCorner=${this.menuAlign === 'start'? 'start-start' : 'start-end'}
.anchorCorner=${this.menuAlign === 'start' ? 'end-start' : 'end-end'}
.menuCorner=${this.menuAlign === 'start' ? 'start-start' : 'start-end'}
@opening=${this.handleOpening}
@opened=${this.redispatchEvent}
@closing=${this.redispatchEvent}
Expand Down Expand Up @@ -515,13 +517,35 @@ export abstract class Select extends selectBaseClass {
const isOpenKey =
event.code === 'Space' ||
event.code === 'ArrowDown' ||
event.code === 'ArrowUp' ||
event.code === 'End' ||
event.code === 'Home' ||
event.code === 'Enter';

// Do not open if currently typing ahead because the user may be typing the
// spacebar to match a word with a space
if (!typeaheadController.isTypingAhead && isOpenKey) {
event.preventDefault();
this.open = true;

// https://www.w3.org/WAI/ARIA/apg/patterns/combobox/examples/combobox-select-only/#kbd_label
switch (event.code) {
case 'Space':
case 'ArrowDown':
case 'Enter':
// We will handle focusing last selected item in this.handleOpening()
this.defaultFocus = FocusState.NONE;
break;
case 'End':
this.defaultFocus = FocusState.LAST_ITEM;
break;
case 'ArrowUp':
case 'Home':
this.defaultFocus = FocusState.FIRST_ITEM;
break;
default:
break;
}
return;
}

Expand Down Expand Up @@ -634,6 +658,12 @@ export abstract class Select extends selectBaseClass {
this.labelEl?.removeAttribute?.('aria-live');
this.redispatchEvent(e);

// FocusState.NONE means we want to handle focus ourselves and focus the
// last selected item.
if (this.defaultFocus !== FocusState.NONE) {
return;
}

const items = this.menu!.items as SelectOption[];
const activeItem = getActiveItem(items)?.item;
let [selectedItem] = this.lastSelectedOptionRecords[0] ?? [null];
Expand Down

0 comments on commit 8912019

Please sign in to comment.