@@ -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 HeadContribution /* */ = 0b010 ;
690
+ const BodyContribution /* */ = 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,14 @@ const clientRenderedSuspenseBoundaryError1D =
4005
4054
const clientRenderedSuspenseBoundaryError2 =
4006
4055
stringToPrecomputedChunk ( '></template>' ) ;
4007
4056
4057
+ const boundaryPreambleContributionChunkTotal =
4058
+ stringToPrecomputedChunk ( '<!--P-->' ) ;
4059
+ const boundaryPreambleContributionChunkStart =
4060
+ stringToPrecomputedChunk ( '<!--P' ) ;
4061
+ const ContributionChunk = stringToPrecomputedChunk ( '1' ) ;
4062
+ const NonContributionChunk = stringToPrecomputedChunk ( ' ' ) ;
4063
+ const boundaryPreambleContributionChunkEnd = stringToPrecomputedChunk ( '-->' ) ;
4064
+
4008
4065
export function writeStartCompletedSuspenseBoundary (
4009
4066
destination : Destination ,
4010
4067
renderState : RenderState ,
@@ -4091,7 +4148,11 @@ export function writeStartClientRenderedSuspenseBoundary(
4091
4148
export function writeEndCompletedSuspenseBoundary (
4092
4149
destination : Destination ,
4093
4150
renderState : RenderState ,
4151
+ preambleState : null | PreambleState ,
4094
4152
) : boolean {
4153
+ if ( preambleState ) {
4154
+ writePreambleContribution ( destination , preambleState ) ;
4155
+ }
4095
4156
return writeChunkAndReturn ( destination , endSuspenseBoundary ) ;
4096
4157
}
4097
4158
export function writeEndPendingSuspenseBoundary (
@@ -4103,9 +4164,47 @@ export function writeEndPendingSuspenseBoundary(
4103
4164
export function writeEndClientRenderedSuspenseBoundary (
4104
4165
destination : Destination ,
4105
4166
renderState : RenderState ,
4167
+ preambleState : null | PreambleState ,
4106
4168
) : boolean {
4169
+ if ( preambleState ) {
4170
+ writePreambleContribution ( destination , preambleState ) ;
4171
+ }
4107
4172
return writeChunkAndReturn ( destination , endSuspenseBoundary ) ;
4108
4173
}
4174
+ function writePreambleContribution (
4175
+ destination : Destination ,
4176
+ preambleState : PreambleState ,
4177
+ ) {
4178
+ const contribution = preambleState . contribution ;
4179
+ if ( contribution !== NoContribution ) {
4180
+ if ( contribution === TotalContribution ) {
4181
+ // We shortcut the preamble marker for boundaries that contribute the entire preamble
4182
+ // since we expect this to be the most common case
4183
+ writeChunk ( destination , boundaryPreambleContributionChunkTotal ) ;
4184
+ } else {
4185
+ writeChunk ( destination , boundaryPreambleContributionChunkStart ) ;
4186
+ writeChunk (
4187
+ destination ,
4188
+ contribution & HTMLContribution
4189
+ ? ContributionChunk
4190
+ : NonContributionChunk ,
4191
+ ) ;
4192
+ writeChunk (
4193
+ destination ,
4194
+ contribution & HeadContribution
4195
+ ? ContributionChunk
4196
+ : NonContributionChunk ,
4197
+ ) ;
4198
+ writeChunk (
4199
+ destination ,
4200
+ contribution & BodyContribution
4201
+ ? ContributionChunk
4202
+ : NonContributionChunk ,
4203
+ ) ;
4204
+ writeChunk ( destination , boundaryPreambleContributionChunkEnd ) ;
4205
+ }
4206
+ }
4207
+ }
4109
4208
4110
4209
const startSegmentHTML = stringToPrecomputedChunk ( '<div hidden id="' ) ;
4111
4210
const startSegmentHTML2 = stringToPrecomputedChunk ( '">' ) ;
0 commit comments