From 536f79a510c11a7473f73b4a121c7b3396ee3d46 Mon Sep 17 00:00:00 2001 From: Kristiyan Kostadinov Date: Thu, 14 Nov 2024 12:40:38 +0100 Subject: [PATCH 1/2] fix(material/form-field): avoid touching the DOM on each state change Currently we set the `aria-describedby` every time the state of the form control changes. This is excessive, because it only needs to happen if the error state or `userAriaDescribedBy` change. --- src/material/form-field/form-field.ts | 18 ++++++++++++++++-- src/material/input/input.ts | 6 ++++-- 2 files changed, 20 insertions(+), 4 deletions(-) diff --git a/src/material/form-field/form-field.ts b/src/material/form-field/form-field.ts index 8b20dc2ccd02..c47904e7faa0 100644 --- a/src/material/form-field/form-field.ts +++ b/src/material/form-field/form-field.ts @@ -36,7 +36,7 @@ import {AbstractControlDirective} from '@angular/forms'; import {ThemePalette} from '@angular/material/core'; import {_IdGenerator} from '@angular/cdk/a11y'; import {Subject, Subscription, merge} from 'rxjs'; -import {takeUntil} from 'rxjs/operators'; +import {map, pairwise, takeUntil, filter, startWith} from 'rxjs/operators'; import {MAT_ERROR, MatError} from './directives/error'; import { FLOATING_LABEL_PARENT, @@ -328,6 +328,7 @@ export class MatFormField private _previousControl: MatFormFieldControl | null = null; private _stateChanges: Subscription | undefined; private _valueChanges: Subscription | undefined; + private _describedByChanges: Subscription | undefined; private _injector = inject(Injector); @@ -377,6 +378,7 @@ export class MatFormField ngOnDestroy() { this._stateChanges?.unsubscribe(); this._valueChanges?.unsubscribe(); + this._describedByChanges?.unsubscribe(); this._destroyed.next(); this._destroyed.complete(); } @@ -426,10 +428,22 @@ export class MatFormField this._stateChanges?.unsubscribe(); this._stateChanges = control.stateChanges.subscribe(() => { this._updateFocusState(); - this._syncDescribedByIds(); this._changeDetectorRef.markForCheck(); }); + // Updating the `aria-describedby` touches the DOM. Only do it if it actually needs to change. + this._describedByChanges?.unsubscribe(); + this._describedByChanges = control.stateChanges + .pipe( + startWith([undefined, undefined] as const), + map(() => [control.errorState, control.userAriaDescribedBy] as const), + pairwise(), + filter(([[prevErrorState, prevDescribedBy], [currentErrorState, currentDescribedBy]]) => { + return prevErrorState !== currentErrorState || prevDescribedBy !== currentDescribedBy; + }), + ) + .subscribe(() => this._syncDescribedByIds()); + this._valueChanges?.unsubscribe(); // Run change detection if the value changes. diff --git a/src/material/input/input.ts b/src/material/input/input.ts index 568b49c6e7bf..a1914bb91967 100644 --- a/src/material/input/input.ts +++ b/src/material/input/input.ts @@ -551,10 +551,12 @@ export class MatInput * @docs-private */ setDescribedByIds(ids: string[]) { + const element = this._elementRef.nativeElement; + if (ids.length) { - this._elementRef.nativeElement.setAttribute('aria-describedby', ids.join(' ')); + element.setAttribute('aria-describedby', ids.join(' ')); } else { - this._elementRef.nativeElement.removeAttribute('aria-describedby'); + element.removeAttribute('aria-describedby'); } } From 7f9a3282018e1455e08ffa63d36999746f4d9643 Mon Sep 17 00:00:00 2001 From: Kristiyan Kostadinov Date: Thu, 14 Nov 2024 13:36:19 +0100 Subject: [PATCH 2/2] fix(material/input): preserve aria-describedby set externally Currently there are two sources of an `aria-describedby` for a `matInput`: the IDs of the hint/error message in the form field and any custom ones set through `aria-describedby`. This is insufficient, because the ID can also come from a direct DOM manupulation like in the `AriaDescriber`. These changes tweak the logic to try and preserve them, because currently they get overwritten. Fixes #30011. --- src/material/input/input.spec.ts | 12 ++++++++++++ src/material/input/input.ts | 24 ++++++++++++++++++++++-- 2 files changed, 34 insertions(+), 2 deletions(-) diff --git a/src/material/input/input.spec.ts b/src/material/input/input.spec.ts index 93ed2c7a723a..611cdc3320f2 100644 --- a/src/material/input/input.spec.ts +++ b/src/material/input/input.spec.ts @@ -643,6 +643,18 @@ describe('MatMdcInput without forms', () => { expect(input.getAttribute('aria-describedby')).toBe('start end'); })); + it('should preserve aria-describedby set directly in the DOM', fakeAsync(() => { + const fixture = createComponent(MatInputHintLabel2TestController); + const input = fixture.nativeElement.querySelector('input'); + input.setAttribute('aria-describedby', 'custom'); + fixture.componentInstance.label = 'label'; + fixture.changeDetectorRef.markForCheck(); + fixture.detectChanges(); + const hint = fixture.nativeElement.querySelector('.mat-mdc-form-field-hint'); + + expect(input.getAttribute('aria-describedby')).toBe(`${hint.getAttribute('id')} custom`); + })); + it('should set a class on the hint element based on its alignment', fakeAsync(() => { const fixture = createComponent(MatInputMultipleHintTestController); diff --git a/src/material/input/input.ts b/src/material/input/input.ts index a1914bb91967..3be8e80a1ebe 100644 --- a/src/material/input/input.ts +++ b/src/material/input/input.ts @@ -111,6 +111,9 @@ export class MatInput private _webkitBlinkWheelListenerAttached = false; private _config = inject(MAT_INPUT_CONFIG, {optional: true}); + /** `aria-describedby` IDs assigned by the form field. */ + private _formFieldDescribedBy: string[] | undefined; + /** Whether the component is being rendered on the server. */ readonly _isServer: boolean; @@ -552,9 +555,26 @@ export class MatInput */ setDescribedByIds(ids: string[]) { const element = this._elementRef.nativeElement; + const existingDescribedBy = element.getAttribute('aria-describedby'); + let toAssign: string[]; + + // In some cases there might be some `aria-describedby` IDs that were assigned directly, + // like by the `AriaDescriber` (see #30011). Attempt to preserve them by taking the previous + // attribute value and filtering out the IDs that came from the previous `setDescribedByIds` + // call. Note the `|| ids` here allows us to avoid duplicating IDs on the first render. + if (existingDescribedBy) { + const exclude = this._formFieldDescribedBy || ids; + toAssign = ids.concat( + existingDescribedBy.split(' ').filter(id => id && !exclude.includes(id)), + ); + } else { + toAssign = ids; + } + + this._formFieldDescribedBy = ids; - if (ids.length) { - element.setAttribute('aria-describedby', ids.join(' ')); + if (toAssign.length) { + element.setAttribute('aria-describedby', toAssign.join(' ')); } else { element.removeAttribute('aria-describedby'); }