From 0897f49d7538ea1c0b3a59a25b60a59cd0208b2d Mon Sep 17 00:00:00 2001 From: Artur Androsovych Date: Thu, 17 Mar 2022 20:36:34 +0200 Subject: [PATCH] fix(material/autocomplete): re-enter the Angular zone when the `NgZone.onStable` emits (#24569) The `NgZone.onStable` always emits outside of the Angular zone, but the zone has not been re-entered. This leads to change detection being called outside of the Angular zone and the `autocomplete.opened` also was emitting values outside of the Angular zone. (cherry picked from commit ff0119648b4a65c501df2277414ff2be7aff8028) --- .../autocomplete/autocomplete-trigger.ts | 35 +++++++++++-------- 1 file changed, 20 insertions(+), 15 deletions(-) diff --git a/src/material/autocomplete/autocomplete-trigger.ts b/src/material/autocomplete/autocomplete-trigger.ts index 9d711eb11c30..b46a2cafe29b 100644 --- a/src/material/autocomplete/autocomplete-trigger.ts +++ b/src/material/autocomplete/autocomplete-trigger.ts @@ -506,22 +506,27 @@ export abstract class _MatAutocompleteTriggerBase // create a new stream of panelClosingActions, replacing any previous streams // that were created, and flatten it so our stream only emits closing events... switchMap(() => { - const wasOpen = this.panelOpen; - this._resetActiveItem(); - this.autocomplete._setVisibility(); - this._changeDetectorRef.detectChanges(); - - if (this.panelOpen) { - this._overlayRef!.updatePosition(); - - // If the `panelOpen` state changed, we need to make sure to emit the `opened` - // event, because we may not have emitted it when the panel was attached. This - // can happen if the users opens the panel and there are no options, but the - // options come in slightly later or as a result of the value changing. - if (wasOpen !== this.panelOpen) { - this.autocomplete.opened.emit(); + // The `NgZone.onStable` always emits outside of the Angular zone, thus we have to re-enter + // the Angular zone. This will lead to change detection being called outside of the Angular + // zone and the `autocomplete.opened` will also emit outside of the Angular. + this._zone.run(() => { + const wasOpen = this.panelOpen; + this._resetActiveItem(); + this.autocomplete._setVisibility(); + this._changeDetectorRef.detectChanges(); + + if (this.panelOpen) { + this._overlayRef!.updatePosition(); + + // If the `panelOpen` state changed, we need to make sure to emit the `opened` + // event, because we may not have emitted it when the panel was attached. This + // can happen if the users opens the panel and there are no options, but the + // options come in slightly later or as a result of the value changing. + if (wasOpen !== this.panelOpen) { + this.autocomplete.opened.emit(); + } } - } + }); return this.panelClosingActions; }),