@@ -58,10 +58,10 @@ import {
58
58
Hydrating ,
59
59
ContentReset ,
60
60
DidCapture ,
61
+ Update ,
61
62
Ref ,
62
63
Deletion ,
63
64
ForceUpdateForLegacySuspense ,
64
- StaticMask ,
65
65
} from './ReactFiberFlags' ;
66
66
import ReactSharedInternals from 'shared/ReactSharedInternals' ;
67
67
import {
@@ -671,6 +671,8 @@ function updateProfiler(
671
671
renderLanes : Lanes ,
672
672
) {
673
673
if ( enableProfilerTimer ) {
674
+ workInProgress . flags |= Update ;
675
+
674
676
// Reset effect durations for the next eventual effect phase.
675
677
// These are reset during render to allow the DevTools commit hook a chance to read them,
676
678
const stateNode = workInProgress . stateNode ;
@@ -1077,9 +1079,6 @@ function updateHostComponent(
1077
1079
workInProgress . flags |= ContentReset ;
1078
1080
}
1079
1081
1080
- // React DevTools reads this flag.
1081
- workInProgress . flags |= PerformedWork ;
1082
-
1083
1082
markRef ( current , workInProgress ) ;
1084
1083
reconcileChildren ( current , workInProgress , nextChildren , renderLanes ) ;
1085
1084
return workInProgress . child ;
@@ -2005,14 +2004,9 @@ function updateSuspensePrimaryChildren(
2005
2004
primaryChildFragment . sibling = null ;
2006
2005
if ( currentFallbackChildFragment !== null ) {
2007
2006
// Delete the fallback child fragment
2008
- const deletions = workInProgress . deletions ;
2009
- if ( deletions === null ) {
2010
- workInProgress . deletions = [ currentFallbackChildFragment ] ;
2011
- // TODO (effects) Rename this to better reflect its new usage (e.g. ChildDeletions)
2012
- workInProgress . flags |= Deletion ;
2013
- } else {
2014
- deletions . push ( currentFallbackChildFragment ) ;
2015
- }
2007
+ currentFallbackChildFragment . nextEffect = null ;
2008
+ currentFallbackChildFragment . flags = Deletion ;
2009
+ workInProgress . firstEffect = workInProgress . lastEffect = currentFallbackChildFragment ;
2016
2010
}
2017
2011
2018
2012
workInProgress . child = primaryChildFragment ;
@@ -2069,19 +2063,24 @@ function updateSuspenseFallbackChildren(
2069
2063
2070
2064
// The fallback fiber was added as a deletion effect during the first pass.
2071
2065
// However, since we're going to remain on the fallback, we no longer want
2072
- // to delete it.
2073
- workInProgress . deletions = null ;
2066
+ // to delete it. So we need to remove it from the list. Deletions are stored
2067
+ // on the same list as effects. We want to keep the effects from the primary
2068
+ // tree. So we copy the primary child fragment's effect list, which does not
2069
+ // include the fallback deletion effect.
2070
+ const progressedLastEffect = primaryChildFragment . lastEffect ;
2071
+ if ( progressedLastEffect !== null ) {
2072
+ workInProgress . firstEffect = primaryChildFragment . firstEffect ;
2073
+ workInProgress . lastEffect = progressedLastEffect ;
2074
+ progressedLastEffect . nextEffect = null ;
2075
+ } else {
2076
+ // TODO: Reset this somewhere else? Lol legacy mode is so weird.
2077
+ workInProgress . firstEffect = workInProgress . lastEffect = null ;
2078
+ }
2074
2079
} else {
2075
2080
primaryChildFragment = createWorkInProgressOffscreenFiber (
2076
2081
currentPrimaryChildFragment ,
2077
2082
primaryChildProps ,
2078
2083
) ;
2079
-
2080
- // Since we're reusing a current tree, we need to reuse the flags, too.
2081
- // (We don't do this in legacy mode, because in legacy mode we don't re-use
2082
- // the current tree; see previous branch.)
2083
- primaryChildFragment . subtreeFlags =
2084
- currentPrimaryChildFragment . subtreeFlags & StaticMask ;
2085
2084
}
2086
2085
let fallbackChildFragment ;
2087
2086
if ( currentFallbackChildFragment !== null ) {
@@ -2566,6 +2565,7 @@ function initSuspenseListRenderState(
2566
2565
tail : null | Fiber ,
2567
2566
lastContentRow : null | Fiber ,
2568
2567
tailMode : SuspenseListTailMode ,
2568
+ lastEffectBeforeRendering : null | Fiber ,
2569
2569
) : void {
2570
2570
const renderState : null | SuspenseListRenderState =
2571
2571
workInProgress . memoizedState ;
@@ -2577,6 +2577,7 @@ function initSuspenseListRenderState(
2577
2577
last : lastContentRow ,
2578
2578
tail : tail ,
2579
2579
tailMode : tailMode ,
2580
+ lastEffect : lastEffectBeforeRendering ,
2580
2581
} : SuspenseListRenderState ) ;
2581
2582
} else {
2582
2583
// We can reuse the existing object from previous renders.
@@ -2586,6 +2587,7 @@ function initSuspenseListRenderState(
2586
2587
renderState . last = lastContentRow ;
2587
2588
renderState . tail = tail ;
2588
2589
renderState . tailMode = tailMode ;
2590
+ renderState . lastEffect = lastEffectBeforeRendering ;
2589
2591
}
2590
2592
}
2591
2593
@@ -2667,6 +2669,7 @@ function updateSuspenseListComponent(
2667
2669
tail ,
2668
2670
lastContentRow ,
2669
2671
tailMode ,
2672
+ workInProgress . lastEffect ,
2670
2673
) ;
2671
2674
break ;
2672
2675
}
@@ -2698,6 +2701,7 @@ function updateSuspenseListComponent(
2698
2701
tail ,
2699
2702
null , // last
2700
2703
tailMode ,
2704
+ workInProgress . lastEffect ,
2701
2705
) ;
2702
2706
break ;
2703
2707
}
@@ -2708,6 +2712,7 @@ function updateSuspenseListComponent(
2708
2712
null , // tail
2709
2713
null , // last
2710
2714
undefined ,
2715
+ workInProgress . lastEffect ,
2711
2716
) ;
2712
2717
break ;
2713
2718
}
@@ -2967,14 +2972,15 @@ function remountFiber(
2967
2972
2968
2973
// Delete the old fiber and place the new one.
2969
2974
// Since the old fiber is disconnected, we have to schedule it manually.
2970
- const deletions = returnFiber . deletions ;
2971
- if ( deletions === null ) {
2972
- returnFiber . deletions = [ current ] ;
2973
- // TODO (effects) Rename this to better reflect its new usage (e.g. ChildDeletions)
2974
- returnFiber . flags |= Deletion ;
2975
+ const last = returnFiber . lastEffect ;
2976
+ if ( last !== null ) {
2977
+ last . nextEffect = current ;
2978
+ returnFiber . lastEffect = current ;
2975
2979
} else {
2976
- deletions . push ( current ) ;
2980
+ returnFiber . firstEffect = returnFiber . lastEffect = current ;
2977
2981
}
2982
+ current . nextEffect = null ;
2983
+ current . flags = Deletion ;
2978
2984
2979
2985
newWorkInProgress . flags |= Placement ;
2980
2986
@@ -3059,6 +3065,15 @@ function beginWork(
3059
3065
}
3060
3066
case Profiler :
3061
3067
if ( enableProfilerTimer ) {
3068
+ // Profiler should only call onRender when one of its descendants actually rendered.
3069
+ const hasChildWork = includesSomeLane (
3070
+ renderLanes ,
3071
+ workInProgress . childLanes ,
3072
+ ) ;
3073
+ if ( hasChildWork ) {
3074
+ workInProgress . flags |= Update ;
3075
+ }
3076
+
3062
3077
// Reset effect durations for the next eventual effect phase.
3063
3078
// These are reset during render to allow the DevTools commit hook a chance to read them,
3064
3079
const stateNode = workInProgress . stateNode ;
@@ -3165,6 +3180,7 @@ function beginWork(
3165
3180
// update in the past but didn't complete it.
3166
3181
renderState . rendering = null ;
3167
3182
renderState . tail = null ;
3183
+ renderState . lastEffect = null ;
3168
3184
}
3169
3185
pushSuspenseContext ( workInProgress , suspenseStackCursor . current ) ;
3170
3186
0 commit comments