diff --git a/cspell.json b/cspell.json index 48394a5c..4b0b15a4 100644 --- a/cspell.json +++ b/cspell.json @@ -11,6 +11,8 @@ "autoplay", "bezier", "boxicons", + "callout", + "callouts", "chatbubble", "checkmark", "claviska", @@ -36,6 +38,7 @@ "enterkeyhint", "eqeqeq", "erroneou", + "errormessage", "esbuild", "exportparts", "fieldsets", @@ -120,6 +123,7 @@ "treeitem", "Triaging", "turbolinks", + "typeof", "unbundles", "unbundling", "unicons", diff --git a/docs/assets/plugins/metadata/metadata.js b/docs/assets/plugins/metadata/metadata.js index c061ac0f..fe9f6c2c 100644 --- a/docs/assets/plugins/metadata/metadata.js +++ b/docs/assets/plugins/metadata/metadata.js @@ -117,7 +117,7 @@ method.parameters?.length ? ` ${escapeHtml( - method.parameters.map(param => `${param.name}: ${param.type.text}`).join(', ') + method.parameters.map(param => `${param.name}: ${param.type?.text || ''}`).join(', ') )} ` : '-' diff --git a/docs/components/radio-button.md b/docs/components/radio-button.md index df6de475..03046ef0 100644 --- a/docs/components/radio-button.md +++ b/docs/components/radio-button.md @@ -7,10 +7,10 @@ Radios buttons allow the user to select a single option from a group using a but Radio buttons are designed to be used with [radio groups](/components/radio-group). When a radio button has focus, the arrow keys can be used to change the selected option just like standard radio controls. ```html preview - - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3 ``` @@ -18,31 +18,25 @@ Radio buttons are designed to be used with [radio groups](/components/radio-grou import { SlRadioButton, SlRadioGroup } from '@shoelace-style/shoelace/dist/react'; const App = () => ( - - - Option 1 - - - Option 2 - - - Option 3 - + + Option 1 + Option 2 + Option 3 ); ``` ## Examples -### Checked +### Checked States -To set the initial checked state, use the `checked` attribute. +To set the initial value and checked state, use the `value` attribute on the containing radio group. ```html preview - - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3 ``` @@ -50,16 +44,10 @@ To set the initial checked state, use the `checked` attribute. import { SlRadioButton, SlRadioGroup } from '@shoelace-style/shoelace/dist/react'; const App = () => ( - - - Option 1 - - - Option 2 - - - Option 3 - + + Option 1 + Option 2 + Option 3 ); ``` @@ -69,10 +57,10 @@ const App = () => ( Use the `disabled` attribute to disable a radio button. ```html preview - - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3 ``` @@ -80,14 +68,10 @@ Use the `disabled` attribute to disable a radio button. import { SlRadioButton, SlRadioGroup } from '@shoelace-style/shoelace/dist/react'; const App = () => ( - - - Option 1 - - - Option 2 - - + + Option 1 + Option 2 + Option 3 @@ -99,26 +83,26 @@ const App = () => ( Use the `size` attribute to change a radio button's size. ```html preview - - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3
- - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3
- - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3 ``` @@ -126,26 +110,26 @@ Use the `size` attribute to change a radio button's size. import { SlRadioButton, SlRadioGroup } from '@shoelace-style/shoelace/dist/react'; const App = () => ( - - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3
- - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3
- - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3 ); ``` @@ -155,26 +139,26 @@ const App = () => ( Use the `pill` attribute to give radio buttons rounded edges. ```html preview - - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3
- - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3
- - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3 ``` @@ -182,26 +166,26 @@ Use the `pill` attribute to give radio buttons rounded edges. import { SlRadioButton, SlRadioGroup } from '@shoelace-style/shoelace/dist/react'; const App = () => ( - - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3
- - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3
- - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3 ); ``` @@ -211,18 +195,18 @@ const App = () => ( Use the `prefix` and `suffix` slots to add icons. ```html preview - - + + Option 1 - + Option 2 - + Option 3 @@ -234,18 +218,18 @@ Use the `prefix` and `suffix` slots to add icons. import { SlIcon, SlRadioButton, SlRadioGroup } from '@shoelace-style/shoelace/dist/react'; const App = () => ( - - + + Option 1 - + Option 2 - + Option 3 @@ -259,24 +243,24 @@ const App = () => ( You can omit button labels and use icons instead. Make sure to set a `label` attribute on each icon so screen readers will announce each option correctly. ```html preview - - + + - + - + - + - + @@ -286,101 +270,28 @@ You can omit button labels and use icons instead. Make sure to set a `label` att import { SlIcon, SlRadioButton, SlRadioGroup } from '@shoelace-style/shoelace/dist/react'; const App = () => ( - - + + - + - + - + - + ); ``` -### Custom Validity - -Use the `setCustomValidity()` method to set a custom validation message. This will prevent the form from submitting and make the browser display the error message you provide. To clear the error, call this function with an empty string. - -```html preview -
- - Not me - Me neither - Choose me - -
- Submit -
- -``` - -```jsx react -import { useEffect, useRef } from 'react'; -import { SlButton, SlIcon, SlRadioButton, SlRadioGroup } from '@shoelace-style/shoelace/dist/react'; -const App = () => { - const radio = useRef(null); - const errorMessage = 'You must choose this option'; - function handleChange(event) { - radio.current.setCustomValidity(radio.current.checked ? '' : errorMessage); - } - function handleSubmit(event) { - event.preventDefault(); - alert('All fields are valid!'); - } - useEffect(() => { - radio.current.setCustomValidity(errorMessage); - }, []); - return ( -
- - - Not me - - - Me neither - - - Choose me - - -
- - Submit - -
- ); -}; -``` - [component-metadata:sl-radio-button] diff --git a/docs/components/radio-group.md b/docs/components/radio-group.md index cc0c48ed..9c1bf9a5 100644 --- a/docs/components/radio-group.md +++ b/docs/components/radio-group.md @@ -5,7 +5,7 @@ Radio groups are used to group multiple [radios](/components/radio) or [radio buttons](/components/radio-button) so they function as a single form control. ```html preview - + Option 1 Option 2 Option 3 @@ -16,7 +16,7 @@ Radio groups are used to group multiple [radios](/components/radio) or [radio bu import { SlRadio, SlRadioGroup } from '@shoelace-style/shoelace/dist/react'; const App = () => ( - + Option 1 Option 2 Option 3 @@ -31,10 +31,10 @@ const App = () => ( You can show the fieldset and legend that wraps the radio group using the `fieldset` attribute. If you don't use this option, you should still provide a label so screen readers announce the control correctly. ```html preview - - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3 ``` @@ -42,16 +42,10 @@ You can show the fieldset and legend that wraps the radio group using the `field import { SlRadio, SlRadioGroup } from '@shoelace-style/shoelace/dist/react'; const App = () => ( - - - Option 1 - - - Option 2 - - - Option 3 - + + Option 1 + Option 2 + Option 3 ); ``` @@ -61,10 +55,10 @@ const App = () => ( [Radio buttons](/components/radio-button) offer an alternate way to display radio controls. In this case, an internal [button group](/components/button-group) is used to group the buttons into a single, cohesive control. ```html preview - - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3 ``` @@ -72,16 +66,10 @@ const App = () => ( import { SlRadioButton, SlRadioGroup } from '@shoelace-style/shoelace/dist/react'; const App = () => ( - - - Option 1 - - - Option 2 - - - Option 3 - + + Option 1 + Option 2 + Option 3 ); ``` @@ -92,10 +80,10 @@ Setting the `required` attribute to make selecting an option mandatory. If a val ```html preview
- - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3
Submit @@ -122,14 +110,14 @@ const App = () => { return ( - - + + Option 1 - + Option 2 - + Option 3 @@ -148,10 +136,10 @@ Use the `setCustomValidity()` method to set a custom validation message. This wi ```html preview - - Not me - Me neither - Choose me + + Not me + Me neither + Choose me
Submit @@ -203,16 +191,10 @@ const App = () => { return ( - - - Not me - - - Me neither - - - Choose me - + + Not me + Me neither + Choose me
diff --git a/docs/components/radio.md b/docs/components/radio.md index 25b481ed..ef7497ba 100644 --- a/docs/components/radio.md +++ b/docs/components/radio.md @@ -7,10 +7,10 @@ Radios allow the user to select a single option from a group. Radios are designed to be used with [radio groups](/components/radio-group). ```html preview - - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3 ``` @@ -18,16 +18,10 @@ Radios are designed to be used with [radio groups](/components/radio-group). import { SlRadio, SlRadioGroup } from '@shoelace-style/shoelace/dist/react'; const App = () => ( - - - Option 1 - - - Option 2 - - - Option 3 - + + Option 1 + Option 2 + Option 3 ); ``` @@ -36,15 +30,15 @@ const App = () => ( ## Examples -### Checked +### Initial Value -To set the initial checked state, use the `checked` attribute. +To set the initial value and checked state, use the `value` attribute on the containing radio group. ```html preview - - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3 ``` @@ -52,16 +46,10 @@ To set the initial checked state, use the `checked` attribute. import { SlRadio, SlRadioGroup } from '@shoelace-style/shoelace/dist/react'; const App = () => ( - - - Option 1 - - - Option 2 - - - Option 3 - + + Option 1 + Option 2 + Option 3 ); ``` @@ -71,10 +59,10 @@ const App = () => ( Use the `disabled` attribute to disable a radio. ```html preview - - Option 1 - Option 2 - Option 3 + + Option 1 + Option 2 + Option 3 ``` @@ -82,91 +70,14 @@ Use the `disabled` attribute to disable a radio. import { SlRadio, SlRadioGroup } from '@shoelace-style/shoelace/dist/react'; const App = () => ( - - - Option 1 - - - Option 2 - - + + Option 1 + Option 2 + Option 3 ); ``` -### Custom Validity - -Use the `setCustomValidity()` method to set a custom validation message. This will prevent the form from submitting and make the browser display the error message you provide. To clear the error, call this function with an empty string. - -```html preview - - - Not me - Me neither - Choose me - -
- Submit - - -``` - -```jsx react -import { useEffect, useRef } from 'react'; -import { SlButton, SlIcon, SlRadio, SlRadioGroup } from '@shoelace-style/shoelace/dist/react'; -const App = () => { - const radio = useRef(null); - const errorMessage = 'You must choose this option'; - function handleChange(event) { - radio.current.setCustomValidity(radio.current.checked ? '' : errorMessage); - } - function handleSubmit(event) { - event.preventDefault(); - alert('All fields are valid!'); - } - useEffect(() => { - radio.current.setCustomValidity(errorMessage); - }, []); - return ( -
- - - Not me - - - Me neither - - - Choose me - - -
- - Submit - -
- ); -}; -``` - [component-metadata:sl-radio] diff --git a/docs/resources/changelog.md b/docs/resources/changelog.md index b19b7bd6..31c14ff0 100644 --- a/docs/resources/changelog.md +++ b/docs/resources/changelog.md @@ -10,6 +10,19 @@ _During the beta period, these restrictions may be relaxed in the event of a mis ## Next +This release breaks radio buttons, which is something that needed to happen to solve a longstanding accessibility issue where screen readers announced an incorrect number of radios, e.g. "1 of 1" instead of "1 of 3." Many attempts to solve this without breaking the existing API were made, but none worked across the board. The new implementation upgrades `` to serve as the "form control" while `` and `` serve as options within the form control. + +To upgrade to this version, you will need to rework your radio controls by moving `name` up to the radio group. And instead of setting `checked` to select a specific radio, you can set `value` on the radio group and the checked item will update automatically. + +- 🚨 BREAKING: improved accessibility of `` and `` so they announce properly in all screen readers + - Added the `name` attribute to `` and removed it from `` and `` + - Added the `value` attribute to `` (use this to control which radio is checked) + - Added the `sl-change` event to `sl-radio-group` + - Added `setCustomValidity()` and `reportValidity()` to `` + - Removed the `checked` attribute from `` and `` (use the radio group's `value` attribute instead) + - Removed the `sl-change` event from `` and `` (listen for it on the radio group instead) + - Removed the `invalid` attribute from `` and `` + - Removed `setCustomValidity()` and `reportValidity()` from `` and `` (now available on the radio group) - Revert disabled focus behavior in ``, ``, and `` to be consistent with native form controls and menus [#845](https://github.com/shoelace-style/shoelace/issues/845) ## 2.0.0-beta.79 diff --git a/src/components/radio-button/radio-button.ts b/src/components/radio-button/radio-button.ts index f91833d9..42393bc9 100644 --- a/src/components/radio-button/radio-button.ts +++ b/src/components/radio-button/radio-button.ts @@ -13,15 +13,13 @@ import type { CSSResultGroup } from 'lit'; * @since 2.0 * @status stable * - * @slot - The radio's label. - * - * @event sl-blur - Emitted when the button loses focus. - * @event sl-focus - Emitted when the button gains focus. - * * @slot - The button's label. * @slot prefix - Used to prepend an icon or similar element to the button. * @slot suffix - Used to append an icon or similar element to the button. * + * @event sl-blur - Emitted when the button loses focus. + * @event sl-focus - Emitted when the button gains focus. + * * @csspart base - The component's internal wrapper. * @csspart button - The internal button element. * @csspart prefix - The prefix slot's container. @@ -40,9 +38,6 @@ export default class SlRadioButton extends LitElement { @state() protected hasFocus = false; @state() checked = false; - /** The radio's name attribute. */ - @property({ reflect: true }) name: string; - /** The radio's value attribute. */ @property() value: string; @@ -109,7 +104,6 @@ export default class SlRadioButton extends LitElement { })} aria-disabled=${this.disabled} type="button" - name=${ifDefined(this.name)} value=${ifDefined(this.value)} tabindex="${this.checked ? '0' : '-1'}" @blur=${this.handleBlur} diff --git a/src/components/radio-group/radio-group.ts b/src/components/radio-group/radio-group.ts index 7f39c965..c068b601 100644 --- a/src/components/radio-group/radio-group.ts +++ b/src/components/radio-group/radio-group.ts @@ -40,7 +40,6 @@ export default class SlRadioGroup extends LitElement { @query('.radio-group__validation-input') input: HTMLInputElement; @state() private hasButtonGroup = false; - @state() private isInvalid = false; @state() private errorMessage = ''; @state() private customErrorMessage = ''; @state() private defaultValue = ''; @@ -57,6 +56,12 @@ export default class SlRadioGroup extends LitElement { /** The name assigned to the radio controls. */ @property() name = 'option'; + /** + * This will be true when the control is in an invalid state. Validity is determined by props such as `type`, + * `required`, `minlength`, `maxlength`, and `pattern` using the browser's constraint validation API. + */ + @property({ type: Boolean, reflect: true }) invalid = false; + /** Shows the fieldset and legend that surrounds the radio group. */ @property({ type: Boolean, attribute: 'fieldset', reflect: true }) fieldset = false; @@ -76,16 +81,16 @@ export default class SlRadioGroup extends LitElement { this.defaultValue = this.value; } + /** Sets a custom validation message. If `message` is not empty, the field will be considered invalid. */ setCustomValidity(message = '') { this.customErrorMessage = message; this.errorMessage = message; if (!message) { - this.isInvalid = false; + this.invalid = false; } else { - this.isInvalid = true; + this.invalid = true; this.input.setCustomValidity(message); - this.showNativeErrorMessage(); } } @@ -108,17 +113,18 @@ export default class SlRadioGroup extends LitElement { }; } + /** Checks for validity and shows the browser's validation message if the control is invalid. */ reportValidity(): boolean { const validity = this.validity; this.errorMessage = this.customErrorMessage || validity.valid ? '' : this.input.validationMessage; - this.isInvalid = !validity.valid; + this.invalid = !validity.valid; if (!validity.valid) { this.showNativeErrorMessage(); } - return !this.isInvalid; + return !this.invalid; } private showNativeErrorMessage() { @@ -185,10 +191,7 @@ export default class SlRadioGroup extends LitElement { private handleSlotChange() { const radios = this.getAllRadios(); - radios.forEach(radio => { - radio.name = this.name; - radio.checked = radio.value === this.value; - }); + radios.forEach(radio => (radio.checked = radio.value === this.value)); this.hasButtonGroup = radios.some(radio => radio.tagName.toLowerCase() === 'sl-radio-button'); @@ -230,7 +233,7 @@ export default class SlRadioGroup extends LitElement { part="base" role="radiogroup" aria-errormessage="radio-error-message" - aria-invalid="${this.isInvalid}" + aria-invalid="${this.invalid}" class=${classMap({ 'radio-group': true, 'radio-group--has-fieldset': this.fieldset, diff --git a/src/components/radio/radio.ts b/src/components/radio/radio.ts index 0a4a898c..91dbda0d 100644 --- a/src/components/radio/radio.ts +++ b/src/components/radio/radio.ts @@ -27,9 +27,6 @@ export default class SlRadio extends LitElement { @state() checked = false; @state() protected hasFocus = false; - /** The radio's name attribute. */ - @property({ reflect: true }) name: string; - /** The radio's value attribute. */ @property() value: string; diff --git a/src/components/tab/tab.test.ts b/src/components/tab/tab.test.ts index 460d4feb..f1fda955 100644 --- a/src/components/tab/tab.test.ts +++ b/src/components/tab/tab.test.ts @@ -65,7 +65,7 @@ describe('', () => { }); describe('blur', () => { - it('shoud blur inner div', async () => { + it('should blur inner div', async () => { const el = await fixture(html` Test `); el.focus();