@@ -9,18 +9,18 @@ import {FocusTrap} from '@angular/cdk/a11y';
9
9
import { OverlayRef , OverlaySizeConfig , PositionStrategy } from '@angular/cdk/overlay' ;
10
10
import { TemplatePortal } from '@angular/cdk/portal' ;
11
11
import {
12
- afterRender ,
12
+ afterNextRender ,
13
13
AfterViewInit ,
14
14
Directive ,
15
15
ElementRef ,
16
16
EmbeddedViewRef ,
17
+ inject ,
18
+ ListenerOptions ,
17
19
NgZone ,
18
20
OnDestroy ,
21
+ Renderer2 ,
19
22
TemplateRef ,
20
23
ViewContainerRef ,
21
- inject ,
22
- Renderer2 ,
23
- ListenerOptions ,
24
24
} from '@angular/core' ;
25
25
import { merge , Observable , Subject } from 'rxjs' ;
26
26
import {
@@ -35,8 +35,10 @@ import {
35
35
withLatestFrom ,
36
36
} from 'rxjs/operators' ;
37
37
38
+ import { _bindEventWithOptions } from '@angular/cdk/platform' ;
38
39
import { CELL_SELECTOR , EDIT_PANE_CLASS , EDIT_PANE_SELECTOR , ROW_SELECTOR } from './constants' ;
39
40
import { EditEventDispatcher , HoverContentState } from './edit-event-dispatcher' ;
41
+ import { EditRef } from './edit-ref' ;
40
42
import { EditServices } from './edit-services' ;
41
43
import { FocusDispatcher } from './focus-dispatcher' ;
42
44
import {
@@ -45,8 +47,6 @@ import {
45
47
FocusEscapeNotifierFactory ,
46
48
} from './focus-escape-notifier' ;
47
49
import { closest } from './polyfill' ;
48
- import { EditRef } from './edit-ref' ;
49
- import { _bindEventWithOptions } from '@angular/cdk/platform' ;
50
50
51
51
/**
52
52
* Describes the number of columns before and after the originating cell that the
@@ -61,6 +61,23 @@ export interface CdkPopoverEditColspan {
61
61
/** Used for rate-limiting mousemove events. */
62
62
const MOUSE_MOVE_THROTTLE_TIME_MS = 10 ;
63
63
64
+ function hasRowElement ( nl : NodeList ) {
65
+ for ( let i = 0 ; i < nl . length ; i ++ ) {
66
+ const el = nl [ i ] ;
67
+ if ( ! ( el instanceof HTMLElement ) ) {
68
+ continue ;
69
+ }
70
+ if ( el . matches ( ROW_SELECTOR ) ) {
71
+ return true ;
72
+ }
73
+ }
74
+ return false ;
75
+ }
76
+
77
+ function isRowMutation ( mutation : MutationRecord ) : boolean {
78
+ return hasRowElement ( mutation . addedNodes ) || hasRowElement ( mutation . removedNodes ) ;
79
+ }
80
+
64
81
/**
65
82
* A directive that must be attached to enable editability on a table.
66
83
* It is responsible for setting up delegated event handlers and providing the
@@ -80,11 +97,24 @@ export class CdkEditable implements AfterViewInit, OnDestroy {
80
97
81
98
protected readonly destroyed = new Subject < void > ( ) ;
82
99
83
- private _rendered = new Subject ( ) ;
100
+ private _rowsRendered = new Subject ( ) ;
101
+
102
+ private _rowMutationObserver = globalThis . MutationObserver
103
+ ? new globalThis . MutationObserver ( mutations => {
104
+ if ( mutations . some ( isRowMutation ) ) {
105
+ this . _rowsRendered . next ( ) ;
106
+ }
107
+ } )
108
+ : null ;
84
109
85
110
constructor ( ) {
86
- afterRender ( ( ) => {
87
- this . _rendered . next ( ) ;
111
+ // Keep track of when the rendered rows changes.
112
+ afterNextRender ( ( ) => {
113
+ this . _rowsRendered . next ( ) ;
114
+ this . _rowMutationObserver ?. observe ( this . elementRef . nativeElement , {
115
+ childList : true ,
116
+ subtree : true ,
117
+ } ) ;
88
118
} ) ;
89
119
}
90
120
@@ -95,7 +125,7 @@ export class CdkEditable implements AfterViewInit, OnDestroy {
95
125
ngOnDestroy ( ) : void {
96
126
this . destroyed . next ( ) ;
97
127
this . destroyed . complete ( ) ;
98
- this . _rendered . complete ( ) ;
128
+ this . _rowMutationObserver ?. disconnect ( ) ;
99
129
}
100
130
101
131
private _observableFromEvent < T extends Event > (
@@ -153,9 +183,10 @@ export class CdkEditable implements AfterViewInit, OnDestroy {
153
183
// Keep track of rows within the table. This is used to know which rows with hover content
154
184
// are first or last in the table. They are kept focusable in case focus enters from above
155
185
// or below the table.
156
- this . _rendered
186
+ this . _rowsRendered
157
187
. pipe (
158
188
// Avoid some timing inconsistencies since Angular v19.
189
+ // TODO: see if we can remove this now that we're using MutationObserver.
159
190
debounceTime ( 0 ) ,
160
191
// Optimization: ignore dom changes while focus is within the table as we already
161
192
// ensure that rows above and below the focused/active row are tabbable.
0 commit comments