From a3d589113399d7fd012037291a7daed20f213ce8 Mon Sep 17 00:00:00 2001
From: Davide Mininni
<101575400+DavideMininni-Fincons@users.noreply.github.com>
Date: Tue, 12 Dec 2023 17:37:24 +0100
Subject: [PATCH] fix(sbb-selection-panel): arrow navigation with
radio-button-group with no content should select the radio-button (#2255)
---
.../radio-button-group/radio-button-group.ts | 28 ++++-----
.../selection-panel/selection-panel.e2e.ts | 58 +++++++++++++++++++
.../selection-panel.stories.ts | 31 ++++++++++
3 files changed, 104 insertions(+), 13 deletions(-)
diff --git a/src/components/radio-button/radio-button-group/radio-button-group.ts b/src/components/radio-button/radio-button-group/radio-button-group.ts
index b90c98bce0..5b6e041c19 100644
--- a/src/components/radio-button/radio-button-group/radio-button-group.ts
+++ b/src/components/radio-button/radio-button-group/radio-button-group.ts
@@ -2,7 +2,7 @@ import { CSSResultGroup, html, LitElement, nothing, PropertyValues, TemplateResu
import { customElement, property, state } from 'lit/decorators.js';
import { isArrowKeyPressed, getNextElementIndex, interactivityChecker } from '../../core/a11y';
-import { toggleDatasetEntry, setAttribute } from '../../core/dom';
+import { toggleDatasetEntry, setAttribute, isValidAttribute } from '../../core/dom';
import {
createNamedSlotState,
HandlerRepository,
@@ -241,7 +241,12 @@ export class SbbRadioButtonGroupElement extends LitElement {
}
private _getRadioTabIndex(radio: SbbRadioButtonElement): number {
- return (radio.checked || this._hasSelectionPanel) && !radio.disabled && !this.disabled ? 0 : -1;
+ const isSelected: boolean = radio.checked && !radio.disabled && !this.disabled;
+ const isParentPanelWithContent: boolean =
+ radio.parentElement.nodeName === 'SBB-SELECTION-PANEL' &&
+ isValidAttribute(radio.parentElement, 'data-has-content');
+
+ return isSelected || (this._hasSelectionPanel && isParentPanelWithContent) ? 0 : -1;
}
private _handleKeyDown(evt: KeyboardEvent): void {
@@ -262,17 +267,14 @@ export class SbbRadioButtonGroupElement extends LitElement {
return;
}
- let current: number;
- let nextIndex: number;
-
- if (this._hasSelectionPanel) {
- current = enabledRadios.findIndex((e: SbbRadioButtonElement) => e === evt.target);
- nextIndex = getNextElementIndex(evt, current, enabledRadios.length);
- } else {
- const checked: number = enabledRadios.findIndex(
- (radio: SbbRadioButtonElement) => radio.checked,
- );
- nextIndex = getNextElementIndex(evt, checked, enabledRadios.length);
+ const current: number = enabledRadios.findIndex((e: SbbRadioButtonElement) => e === evt.target);
+ const nextIndex: number = getNextElementIndex(evt, current, enabledRadios.length);
+
+ // Selection on arrow keypress is allowed only if all the selection-panels have no content.
+ const allPanelsHaveNoContent: boolean = (
+ Array.from(this.querySelectorAll?.('sbb-selection-panel')) || []
+ ).every((e) => !isValidAttribute(e, 'data-has-content'));
+ if (!this._hasSelectionPanel || (this._hasSelectionPanel && allPanelsHaveNoContent)) {
enabledRadios[nextIndex].select();
}
diff --git a/src/components/selection-panel/selection-panel.e2e.ts b/src/components/selection-panel/selection-panel.e2e.ts
index 381582680c..8ff7e00370 100644
--- a/src/components/selection-panel/selection-panel.e2e.ts
+++ b/src/components/selection-panel/selection-panel.e2e.ts
@@ -231,6 +231,64 @@ describe('sbb-selection-panel', () => {
});
});
+ describe('with radio group with no slotted content', () => {
+ it('focus selected, the focus and select on keyboard navigation', async () => {
+ await fixture(html`
+
+
+ Value one
+
+
+ Value two
+
+
+ Value three
+
+
+ Value four
+
+
+ `);
+ const wrapperNoContent = document.querySelector('#group-no-content');
+ const firstInputNoContent: SbbRadioButtonElement =
+ document.querySelector('#input-no-content-1');
+ const secondInputNoContent: SbbRadioButtonElement =
+ document.querySelector('#input-no-content-2');
+ const fourthInputNoContent: SbbRadioButtonElement =
+ document.querySelector('#input-no-content-4');
+
+ await sendKeys({ down: 'Tab' });
+ await waitForLitRender(wrapperNoContent);
+ expect(document.activeElement.id).to.be.equal(secondInputNoContent.id);
+
+ await sendKeys({ down: 'ArrowUp' });
+ await waitForLitRender(wrapperNoContent);
+ expect(document.activeElement.id).to.be.equal(firstInputNoContent.id);
+ expect(secondInputNoContent).not.to.have.attribute('checked');
+ expect(firstInputNoContent).to.have.attribute('checked');
+
+ await sendKeys({ down: 'ArrowRight' });
+ await waitForLitRender(wrapperNoContent);
+ expect(document.activeElement.id).to.be.equal(secondInputNoContent.id);
+ expect(firstInputNoContent).not.to.have.attribute('checked');
+ expect(secondInputNoContent).to.have.attribute('checked');
+
+ await sendKeys({ down: 'ArrowDown' });
+ await waitForLitRender(wrapperNoContent);
+ expect(document.activeElement.id).to.be.equal(fourthInputNoContent.id);
+ expect(secondInputNoContent).not.to.have.attribute('checked');
+ expect(fourthInputNoContent).to.have.attribute('checked');
+
+ await sendKeys({ down: 'ArrowLeft' });
+ await waitForLitRender(wrapperNoContent);
+ expect(document.activeElement.id).to.be.equal(secondInputNoContent.id);
+ expect(fourthInputNoContent).not.to.have.attribute('checked');
+ expect(secondInputNoContent).to.have.attribute('checked');
+ });
+ });
+
describe('with nested radio buttons', () => {
let nestedElement: SbbRadioButtonGroupElement;
diff --git a/src/components/selection-panel/selection-panel.stories.ts b/src/components/selection-panel/selection-panel.stories.ts
index fb0e370572..d9217676dc 100644
--- a/src/components/selection-panel/selection-panel.stories.ts
+++ b/src/components/selection-panel/selection-panel.stories.ts
@@ -472,6 +472,31 @@ const WithNoContentTemplate = ({
`;
+const WithNoContentGroupTemplate = ({
+ checkedInput,
+ disabledInput,
+ ...args
+}: Args): TemplateResult => html`
+
+
+ ${cardBadge()}
+
+ Value one ${suffixAndSubtext()}
+
+
+
+ ${cardBadge()}
+
+ Value two ${suffixAndSubtext()}
+
+
+
+ ${cardBadge()}
+ Value three ${suffixAndSubtext()}
+
+
+`;
+
export const WithCheckbox: StoryObj = {
render: WithCheckboxTemplate,
argTypes: basicArgTypes,
@@ -665,6 +690,12 @@ export const WithNoContentCheckedDisabled: StoryObj = {
args: { ...basicArgs, checkedInput: true, disabledInput: true },
};
+export const WithNoContentGroup: StoryObj = {
+ render: WithNoContentGroupTemplate,
+ argTypes: basicArgTypes,
+ args: { ...basicArgs, checkedInput: true },
+};
+
export const TicketsOptionsExample: StoryObj = {
render: TicketsOptionsExampleTemplate,
argTypes: basicArgTypes,