@@ -684,16 +684,25 @@ export function completeResumableState(resumableState: ResumableState): void {
684
684
resumableState . bootstrapModules = undefined ;
685
685
}
686
686
687
+ const NoContribution /* */ = 0b000 ;
688
+ const HTMLContribution /* */ = 0b001 ;
689
+ const BodyContribution /* */ = 0b010 ;
690
+ const HeadContribution /* */ = 0b100 ;
691
+ const TotalContribution =
692
+ HTMLContribution | HeadContribution | BodyContribution ;
693
+
687
694
export type PreambleState = {
688
695
htmlChunks : null | Array < Chunk | PrecomputedChunk > ,
689
696
headChunks : null | Array < Chunk | PrecomputedChunk > ,
690
697
bodyChunks : null | Array < Chunk | PrecomputedChunk > ,
698
+ contribution : number ,
691
699
} ;
692
700
export function createPreambleState ( ) : PreambleState {
693
701
return {
694
702
htmlChunks : null ,
695
703
headChunks : null ,
696
704
bodyChunks : null ,
705
+ contribution : NoContribution ,
697
706
} ;
698
707
}
699
708
@@ -3227,7 +3236,7 @@ function pushStartHead(
3227
3236
throw new Error ( `The ${ '`<head>`' } tag may only be rendered once.` ) ;
3228
3237
}
3229
3238
preamble . headChunks = [ ] ;
3230
- return pushStartGenericElement ( preamble . headChunks , props , 'head' ) ;
3239
+ return pushStartSingletonElement ( preamble . headChunks , props , 'head' ) ;
3231
3240
} else {
3232
3241
// This <head> is deep and is likely just an error. we emit it inline though.
3233
3242
// Validation should warn that this tag is the the wrong spot.
@@ -3251,7 +3260,7 @@ function pushStartBody(
3251
3260
}
3252
3261
3253
3262
preamble . bodyChunks = [ ] ;
3254
- return pushStartGenericElement ( preamble . bodyChunks , props , 'body' ) ;
3263
+ return pushStartSingletonElement ( preamble . bodyChunks , props , 'body' ) ;
3255
3264
} else {
3256
3265
// This <head> is deep and is likely just an error. we emit it inline though.
3257
3266
// Validation should warn that this tag is the the wrong spot.
@@ -3275,7 +3284,7 @@ function pushStartHtml(
3275
3284
}
3276
3285
3277
3286
preamble . htmlChunks = [ DOCTYPE ] ;
3278
- return pushStartGenericElement ( preamble . htmlChunks , props , 'html' ) ;
3287
+ return pushStartSingletonElement ( preamble . htmlChunks , props , 'html' ) ;
3279
3288
} else {
3280
3289
// This <html> is deep and is likely just an error. we emit it inline though.
3281
3290
// Validation should warn that this tag is the the wrong spot.
@@ -3416,6 +3425,43 @@ function pushScriptImpl(
3416
3425
return null ;
3417
3426
}
3418
3427
3428
+ // This is a fork of pushStartGenericElement because we don't ever want to do
3429
+ // the children as strign optimization on that path when rendering singletons.
3430
+ // When we eliminate that special path we can delete this fork and unify it again
3431
+ function pushStartSingletonElement (
3432
+ target : Array < Chunk | PrecomputedChunk > ,
3433
+ props : Object ,
3434
+ tag : string ,
3435
+ ) : ReactNodeList {
3436
+ target . push ( startChunkForTag ( tag ) ) ;
3437
+
3438
+ let children = null ;
3439
+ let innerHTML = null ;
3440
+ for ( const propKey in props ) {
3441
+ if ( hasOwnProperty . call ( props , propKey ) ) {
3442
+ const propValue = props [ propKey ] ;
3443
+ if ( propValue == null ) {
3444
+ continue ;
3445
+ }
3446
+ switch ( propKey ) {
3447
+ case 'children ':
3448
+ children = propValue ;
3449
+ break ;
3450
+ case 'dangerouslySetInnerHTML ':
3451
+ innerHTML = propValue ;
3452
+ break ;
3453
+ default :
3454
+ pushAttribute ( target , propKey , propValue ) ;
3455
+ break ;
3456
+ }
3457
+ }
3458
+ }
3459
+
3460
+ target . push ( endOfStartTag ) ;
3461
+ pushInnerHTML ( target , innerHTML , children ) ;
3462
+ return children ;
3463
+ }
3464
+
3419
3465
function pushStartGenericElement (
3420
3466
target : Array < Chunk | PrecomputedChunk > ,
3421
3467
props : Object ,
@@ -3907,14 +3953,17 @@ export function hoistPreambleState(
3907
3953
preambleState : PreambleState ,
3908
3954
) {
3909
3955
const rootPreamble = renderState . preamble ;
3910
- if ( rootPreamble . htmlChunks === null ) {
3956
+ if ( rootPreamble . htmlChunks === null && preambleState . htmlChunks ) {
3911
3957
rootPreamble . htmlChunks = preambleState . htmlChunks ;
3958
+ preambleState . contribution |= HTMLContribution ;
3912
3959
}
3913
- if ( rootPreamble . headChunks === null ) {
3960
+ if ( rootPreamble . headChunks === null && preambleState . headChunks ) {
3914
3961
rootPreamble . headChunks = preambleState . headChunks ;
3962
+ preambleState . contribution |= HeadContribution ;
3915
3963
}
3916
- if ( rootPreamble . bodyChunks === null ) {
3964
+ if ( rootPreamble . bodyChunks === null && preambleState . bodyChunks ) {
3917
3965
rootPreamble . bodyChunks = preambleState . bodyChunks ;
3966
+ preambleState . contribution |= BodyContribution ;
3918
3967
}
3919
3968
}
3920
3969
@@ -4005,6 +4054,21 @@ const clientRenderedSuspenseBoundaryError1D =
4005
4054
const clientRenderedSuspenseBoundaryError2 =
4006
4055
stringToPrecomputedChunk ( '></template>' ) ;
4007
4056
4057
+ const boundaryPreambleContributionChunkTotal =
4058
+ stringToPrecomputedChunk ( '<!--P-->' ) ;
4059
+ const boundaryPreambleContributionChunkHTMLOnly =
4060
+ stringToPrecomputedChunk ( '<!--P| -->' ) ;
4061
+ const boundaryPreambleContributionChunkBodyOnly =
4062
+ stringToPrecomputedChunk ( '<!--P | -->' ) ;
4063
+ const boundaryPreambleContributionChunkHeadOnly =
4064
+ stringToPrecomputedChunk ( '<!--P -->' ) ;
4065
+ const boundaryPreambleContributionChunkHTMLAndBody =
4066
+ stringToPrecomputedChunk ( '<!--P|| -->' ) ;
4067
+ const boundaryPreambleContributionChunkHTMLAndHead =
4068
+ stringToPrecomputedChunk ( '<!--P| -->' ) ;
4069
+ const boundaryPreambleContributionChunkHeadAndBody =
4070
+ stringToPrecomputedChunk ( '<!--P -->' ) ;
4071
+
4008
4072
export function writeStartCompletedSuspenseBoundary (
4009
4073
destination : Destination ,
4010
4074
renderState : RenderState ,
@@ -4091,7 +4155,11 @@ export function writeStartClientRenderedSuspenseBoundary(
4091
4155
export function writeEndCompletedSuspenseBoundary (
4092
4156
destination : Destination ,
4093
4157
renderState : RenderState ,
4158
+ preambleState : null | PreambleState ,
4094
4159
) : boolean {
4160
+ if ( preambleState ) {
4161
+ writePreambleContribution ( destination , preambleState ) ;
4162
+ }
4095
4163
return writeChunkAndReturn ( destination , endSuspenseBoundary ) ;
4096
4164
}
4097
4165
export function writeEndPendingSuspenseBoundary (
@@ -4103,9 +4171,41 @@ export function writeEndPendingSuspenseBoundary(
4103
4171
export function writeEndClientRenderedSuspenseBoundary (
4104
4172
destination : Destination ,
4105
4173
renderState : RenderState ,
4174
+ preambleState : null | PreambleState ,
4106
4175
) : boolean {
4176
+ if ( preambleState ) {
4177
+ writePreambleContribution ( destination , preambleState ) ;
4178
+ }
4107
4179
return writeChunkAndReturn ( destination , endSuspenseBoundary ) ;
4108
4180
}
4181
+ function writePreambleContribution (
4182
+ destination : Destination ,
4183
+ preambleState : PreambleState ,
4184
+ ) {
4185
+ const contribution = preambleState . contribution ;
4186
+ switch ( contribution ) {
4187
+ case TotalContribution :
4188
+ writeChunk ( destination , boundaryPreambleContributionChunkTotal ) ;
4189
+ break ;
4190
+ case HeadContribution | BodyContribution :
4191
+ writeChunk ( destination , boundaryPreambleContributionChunkHeadAndBody ) ;
4192
+ break ;
4193
+ case HTMLContribution | HeadContribution :
4194
+ writeChunk ( destination , boundaryPreambleContributionChunkHTMLAndHead ) ;
4195
+ break ;
4196
+ case HTMLContribution | BodyContribution :
4197
+ writeChunk ( destination , boundaryPreambleContributionChunkHTMLAndBody ) ;
4198
+ break ;
4199
+ case HeadContribution :
4200
+ writeChunk ( destination , boundaryPreambleContributionChunkHeadOnly ) ;
4201
+ break ;
4202
+ case BodyContribution :
4203
+ writeChunk ( destination , boundaryPreambleContributionChunkBodyOnly ) ;
4204
+ break ;
4205
+ case HTMLContribution :
4206
+ writeChunk ( destination , boundaryPreambleContributionChunkHTMLOnly ) ;
4207
+ }
4208
+ }
4109
4209
4110
4210
const startSegmentHTML = stringToPrecomputedChunk ( '<div hidden id="' ) ;
4111
4211
const startSegmentHTML2 = stringToPrecomputedChunk ( '">' ) ;
0 commit comments