From d910aa2f043f313588f8dc65cee1371141fe9112 Mon Sep 17 00:00:00 2001 From: Nandor_Czegledi Date: Tue, 1 Oct 2024 18:43:39 +0200 Subject: [PATCH] WIP(ui-calendar): add keyboard navigation and focus controll for calendar --- packages/ui-calendar/src/Calendar/index.tsx | 146 +++++++++++++++--- .../__new-tests__/DateInput.test.tsx | 8 +- 2 files changed, 132 insertions(+), 22 deletions(-) diff --git a/packages/ui-calendar/src/Calendar/index.tsx b/packages/ui-calendar/src/Calendar/index.tsx index 6c05fc0394..5ebdad7192 100644 --- a/packages/ui-calendar/src/Calendar/index.tsx +++ b/packages/ui-calendar/src/Calendar/index.tsx @@ -23,7 +23,15 @@ */ /** @jsx jsx */ -import React, { Children, Component, ReactElement, MouseEvent } from 'react' +import React, { + Children, + Component, + ReactElement, + MouseEvent, + KeyboardEvent, + createRef, + FocusEvent +} from 'react' import { View } from '@instructure/ui-view' import { @@ -82,6 +90,9 @@ class Calendar extends Component { role: 'table' } + // Create ref for each calendar day for programmatic focus management + dayRefs: (HTMLElement | null )[] + ref: Element | null = null private _weekdayHeaderIds = ( this.props.renderWeekdayLabels || this.defaultWeekdays @@ -90,18 +101,87 @@ class Calendar extends Component { }, {}) constructor(props: CalendarProps) { super(props) + this.dayRefs = [] + this.state = { + ...this.calculateState(this.locale(), this.timezone(), props.currentDate), + } + } - this.state = this.calculateState( - this.locale(), - this.timezone(), - props.currentDate - ) + setFirstDayTabIndex = (tabIndex: 0 | -1) => { + // Get the first button inside the first calendar day cell and set its tabIndex + if (this.dayRefs[0]) { + this.dayRefs[0].tabIndex = tabIndex + } } handleRef = (el: Element | null) => { this.ref = el } + handleDaysTableBlur = (e: FocusEvent) => { + const dataCid = e.relatedTarget?.getAttribute('data-cid') + + // If the focus leave the days table, + // reset the table focus entry point to its first calendar day + if (dataCid !== 'Day') { + this.setFirstDayTabIndex(0) + } + } + + handleKeyDown = (e: KeyboardEvent, dayIndex: number) => { + const totalDays = Calendar.DAY_COUNT + const daysPerWeek = 7 + let targetIndex: number + + const moveFocus = (targetIndex: number) => { + const targetDay = this.dayRefs[targetIndex] + + targetDay?.focus() + this.setFirstDayTabIndex(-1) + } + + switch (e.key) { + case 'ArrowLeft': + e.preventDefault() + targetIndex = + dayIndex % daysPerWeek === 0 + ? dayIndex + daysPerWeek - 1 + : dayIndex - 1 + moveFocus(targetIndex) + break + + case 'ArrowRight': + e.preventDefault() + targetIndex = + (dayIndex + 1) % daysPerWeek === 0 + ? dayIndex - (daysPerWeek - 1) + : dayIndex + 1 + moveFocus(targetIndex) + break + + case 'ArrowUp': + e.preventDefault() + targetIndex = + dayIndex < daysPerWeek + ? dayIndex + totalDays - daysPerWeek + : dayIndex - daysPerWeek + moveFocus(targetIndex) + break + + case 'ArrowDown': + e.preventDefault() + targetIndex = + dayIndex >= totalDays - daysPerWeek + ? dayIndex - totalDays + daysPerWeek + : dayIndex + daysPerWeek + moveFocus(targetIndex) + break + + default: + break + } + } + componentDidMount() { this.props.makeStyles?.() } @@ -115,8 +195,9 @@ class Calendar extends Component { prevProps.visibleMonth !== this.props.visibleMonth if (isUpdated) { - this.setState(() => { + this.setState((prevState) => { return { + ...prevState, ...this.calculateState( this.locale(), this.timezone(), @@ -322,7 +403,12 @@ class Calendar extends Component { return ( {this.renderWeekdayHeaders()} - {this.renderDays()} + {/* eslint-disable-next-line jsx-a11y/no-noninteractive-element-interactions */} + this.handleDaysTableBlur(e)} + > + {this.renderDays()} +
) } @@ -408,17 +494,28 @@ class Calendar extends Component { days[index].push(day) return days // 7xN 2D array of `Day`s }, []) - .map((row) => ( + .map((row, rowIndex) => ( - {row.map((day, i) => ( - - {role === 'presentation' - ? safeCloneElement(day, { - 'aria-describedby': this._weekdayHeaderIds[i] - }) - : day} - - ))} + {row.map((day, i) => { + const dayIndex = rowIndex * 7 + i + + return ( + // eslint-disable-next-line jsx-a11y/no-noninteractive-element-interactions + + {role === 'presentation' + ? safeCloneElement(day, { + 'aria-describedby': this._weekdayHeaderIds[i], + tabIndex: dayIndex === 0 ? 0 : -1, + }) + : safeCloneElement(day, { + tabIndex: dayIndex === 0 ? 0 : -1, + })} + + ) + })} )) } @@ -441,8 +538,10 @@ class Calendar extends Component { return DateTime.browserTimeZone() } - // date is returned es a ISO string, like 2021-09-14T22:00:00.000Z + // date is returned as a ISO string, like 2021-09-14T22:00:00.000Z handleDayClick = (event: MouseEvent, { date }: { date: string }) => { + this.setFirstDayTabIndex(-1) + if (this.props.onDateSelected) { const parsedDate = DateTime.parse(date, this.locale(), this.timezone()) this.props.onDateSelected(parsedDate.toISOString(), parsedDate, event) @@ -465,6 +564,10 @@ class Calendar extends Component { return disabledDates(date.toISOString()) } + setDayRef = (index: number) => (el: Element | null) => { + this.dayRefs[index] = el as HTMLElement | null + } + renderDefaultdays() { const { selectedDate } = this.props const { visibleMonth, today } = this.state @@ -479,8 +582,9 @@ class Calendar extends Component { arr.push(currDate.clone()) currDate.add({ days: 1 }) } - return arr.map((date) => { + return arr.map((date, dayIndex) => { const dateStr = date.toISOString() + return ( { label={date.format('D MMMM YYYY')} // used by screen readers onClick={this.handleDayClick} interaction={this.isDisabledDate(date) ? 'disabled' : 'enabled'} + elementRef={this.setDayRef(dayIndex)} + onKeyDown={(e) => this.handleKeyDown(e, dayIndex)} > {date.format('DD')} diff --git a/packages/ui-date-input/src/DateInput/__new-tests__/DateInput.test.tsx b/packages/ui-date-input/src/DateInput/__new-tests__/DateInput.test.tsx index f0823e6bbd..8dd92d5aa3 100644 --- a/packages/ui-date-input/src/DateInput/__new-tests__/DateInput.test.tsx +++ b/packages/ui-date-input/src/DateInput/__new-tests__/DateInput.test.tsx @@ -740,8 +740,12 @@ describe('', () => { expect(prevMonthButton).toHaveAttribute('tabIndex', '-1') expect(calendarDays).toHaveLength(42) - calendarDays.forEach((day) => { - expect(day).toHaveAttribute('tabIndex', '-1') + calendarDays.forEach((day, index) => { + if (index === 0) { + expect(day).toHaveAttribute('tabIndex', '0') + } else { + expect(day).toHaveAttribute('tabIndex', '-1') + } }) })