@@ -75,9 +75,10 @@ import {
75
75
Hydrating ,
76
76
HydratingAndUpdate ,
77
77
Passive ,
78
+ BeforeMutationMask ,
78
79
MutationMask ,
79
- PassiveMask ,
80
80
LayoutMask ,
81
+ PassiveMask ,
81
82
PassiveUnmountPendingDev ,
82
83
} from './ReactFiberFlags' ;
83
84
import getComponentName from 'shared/getComponentName' ;
@@ -128,6 +129,8 @@ import {
128
129
commitHydratedSuspenseInstance ,
129
130
clearContainer ,
130
131
prepareScopeUpdate ,
132
+ prepareForCommit ,
133
+ beforeActiveInstanceBlur ,
131
134
} from './ReactFiberHostConfig' ;
132
135
import {
133
136
captureCommitPhaseError ,
@@ -144,6 +147,7 @@ import {
144
147
Passive as HookPassive ,
145
148
} from './ReactHookEffectTags' ;
146
149
import { didWarnAboutReassigningProps } from './ReactFiberBeginWork.new' ;
150
+ import { doesFiberContain } from './ReactFiberTreeReflection' ;
147
151
148
152
let didWarnAboutUndefinedSnapshotBeforeUpdate : Set < mixed > | null = null ;
149
153
if ( __DEV__ ) {
@@ -259,18 +263,114 @@ function safelyCallDestroy(current: Fiber, destroy: () => void) {
259
263
}
260
264
}
261
265
262
- function commitBeforeMutationLifeCycles (
263
- current : Fiber | null ,
264
- finishedWork : Fiber ,
265
- ) : void {
266
- switch ( finishedWork . tag ) {
267
- case FunctionComponent :
268
- case ForwardRef :
269
- case SimpleMemoComponent : {
266
+ let focusedInstanceHandle : null | Fiber = null ;
267
+ let shouldFireAfterActiveInstanceBlur : boolean = false ;
268
+
269
+ export function commitBeforeMutationEffects (
270
+ root : FiberRoot ,
271
+ firstChild : Fiber ,
272
+ ) {
273
+ focusedInstanceHandle = prepareForCommit ( root . containerInfo ) ;
274
+
275
+ nextEffect = firstChild ;
276
+ commitBeforeMutationEffects_begin ( ) ;
277
+
278
+ // We no longer need to track the active instance fiber
279
+ const shouldFire = shouldFireAfterActiveInstanceBlur ;
280
+ shouldFireAfterActiveInstanceBlur = false ;
281
+ focusedInstanceHandle = null ;
282
+
283
+ return shouldFire ;
284
+ }
285
+
286
+ function commitBeforeMutationEffects_begin ( ) {
287
+ while ( nextEffect !== null ) {
288
+ const fiber = nextEffect ;
289
+
290
+ // TODO: Should wrap this in flags check, too, as optimization
291
+ const deletions = fiber . deletions ;
292
+ if ( deletions !== null ) {
293
+ for ( let i = 0 ; i < deletions . length ; i ++ ) {
294
+ const deletion = deletions [ i ] ;
295
+ commitBeforeMutationEffectsDeletion ( deletion ) ;
296
+ }
297
+ }
298
+
299
+ const child = fiber . child ;
300
+ if (
301
+ ( fiber . subtreeFlags & BeforeMutationMask ) !== NoFlags &&
302
+ child !== null
303
+ ) {
304
+ ensureCorrectReturnPointer ( child , fiber ) ;
305
+ nextEffect = child ;
306
+ } else {
307
+ commitBeforeMutationEffects_complete ( ) ;
308
+ }
309
+ }
310
+ }
311
+
312
+ function commitBeforeMutationEffects_complete ( ) {
313
+ while ( nextEffect !== null ) {
314
+ const fiber = nextEffect ;
315
+ if ( __DEV__ ) {
316
+ setCurrentDebugFiberInDEV ( fiber ) ;
317
+ invokeGuardedCallback (
318
+ null ,
319
+ commitBeforeMutationEffectsOnFiber ,
320
+ null ,
321
+ fiber ,
322
+ ) ;
323
+ if ( hasCaughtError ( ) ) {
324
+ const error = clearCaughtError ( ) ;
325
+ captureCommitPhaseError ( fiber , error ) ;
326
+ }
327
+ resetCurrentDebugFiberInDEV ( ) ;
328
+ } else {
329
+ try {
330
+ commitBeforeMutationEffectsOnFiber ( fiber ) ;
331
+ } catch ( error ) {
332
+ captureCommitPhaseError ( fiber , error ) ;
333
+ }
334
+ }
335
+
336
+ const sibling = fiber . sibling ;
337
+ if ( sibling !== null ) {
338
+ ensureCorrectReturnPointer ( sibling , fiber . return ) ;
339
+ nextEffect = sibling ;
270
340
return ;
271
341
}
272
- case ClassComponent : {
273
- if ( finishedWork . flags & Snapshot ) {
342
+
343
+ nextEffect = fiber . return ;
344
+ }
345
+ }
346
+
347
+ function commitBeforeMutationEffectsOnFiber ( finishedWork : Fiber ) {
348
+ const current = finishedWork . alternate ;
349
+ const flags = finishedWork . flags ;
350
+
351
+ if ( ! shouldFireAfterActiveInstanceBlur && focusedInstanceHandle !== null ) {
352
+ // Check to see if the focused element was inside of a hidden (Suspense) subtree.
353
+ // TODO: Move this out of the hot path using a dedicated effect tag.
354
+ if (
355
+ finishedWork . tag === SuspenseComponent &&
356
+ isSuspenseBoundaryBeingHidden ( current , finishedWork ) &&
357
+ doesFiberContain ( finishedWork , focusedInstanceHandle )
358
+ ) {
359
+ shouldFireAfterActiveInstanceBlur = true ;
360
+ beforeActiveInstanceBlur ( finishedWork ) ;
361
+ }
362
+ }
363
+
364
+ if ( ( flags & Snapshot ) !== NoFlags ) {
365
+ setCurrentDebugFiberInDEV ( finishedWork ) ;
366
+
367
+ switch ( finishedWork . tag ) {
368
+ case FunctionComponent :
369
+ case ForwardRef :
370
+ case SimpleMemoComponent : {
371
+ break ;
372
+ }
373
+ case ClassComponent : {
274
374
if ( current !== null ) {
275
375
const prevProps = current . memoizedProps ;
276
376
const prevState = current . memoizedState ;
@@ -324,30 +424,43 @@ function commitBeforeMutationLifeCycles(
324
424
}
325
425
instance . __reactInternalSnapshotBeforeUpdate = snapshot ;
326
426
}
427
+ break ;
327
428
}
328
- return ;
329
- }
330
- case HostRoot : {
331
- if ( supportsMutation ) {
332
- if ( finishedWork . flags & Snapshot ) {
429
+ case HostRoot : {
430
+ if ( supportsMutation ) {
333
431
const root = finishedWork . stateNode ;
334
432
clearContainer ( root . containerInfo ) ;
335
433
}
434
+ break ;
435
+ }
436
+ case HostComponent :
437
+ case HostText :
438
+ case HostPortal :
439
+ case IncompleteClassComponent :
440
+ // Nothing to do for these component types
441
+ break ;
442
+ default : {
443
+ invariant (
444
+ false ,
445
+ 'This unit of work tag should not have side-effects. This error is ' +
446
+ 'likely caused by a bug in React. Please file an issue.' ,
447
+ ) ;
336
448
}
337
- return ;
338
449
}
339
- case HostComponent :
340
- case HostText :
341
- case HostPortal :
342
- case IncompleteClassComponent :
343
- // Nothing to do for these component types
344
- return ;
450
+
451
+ resetCurrentDebugFiberInDEV ( ) ;
452
+ }
453
+ }
454
+
455
+ function commitBeforeMutationEffectsDeletion ( deletion : Fiber ) {
456
+ // TODO (effects) It would be nice to avoid calling doesFiberContain()
457
+ // Maybe we can repurpose one of the subtreeFlags positions for this instead?
458
+ // Use it to store which part of the tree the focused instance is in?
459
+ // This assumes we can safely determine that instance during the "render" phase.
460
+ if ( doesFiberContain ( deletion , ( ( focusedInstanceHandle : any ) : Fiber ) ) ) {
461
+ shouldFireAfterActiveInstanceBlur = true ;
462
+ beforeActiveInstanceBlur ( deletion ) ;
345
463
}
346
- invariant (
347
- false ,
348
- 'This unit of work tag should not have side-effects. This error is ' +
349
- 'likely caused by a bug in React. Please file an issue.' ,
350
- ) ;
351
464
}
352
465
353
466
function commitHookEffectListUnmount ( flags : HookFlags , finishedWork : Fiber ) {
@@ -2353,7 +2466,6 @@ function ensureCorrectReturnPointer(fiber, expectedReturnFiber) {
2353
2466
}
2354
2467
2355
2468
export {
2356
- commitBeforeMutationLifeCycles ,
2357
2469
commitResetTextContent ,
2358
2470
commitPlacement ,
2359
2471
commitDeletion ,
0 commit comments