Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

double invoke test #1

Closed
wants to merge 44 commits into from
Closed
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
44 commits
Select commit Hold shift + click to select a range
fecc288
[react devtools] Device storage support (#25452)
rbalicki2 Oct 25, 2022
1720405
[Float] fix coordination of resource identity and hydration (#25569)
gnoff Oct 27, 2022
09def59
[Float] handle noscript context for Resources (#25559)
gnoff Oct 27, 2022
28a574e
Try assigning fetch to globalThis if global assignment fails (#25571)
sebmarkbage Oct 27, 2022
cf3932b
Remove old react-fetch, react-fs and react-pg libraries (#25577)
sebmarkbage Oct 27, 2022
d2a0176
Detect and warn if use(promise) is wrapped with try/catch block (#25543)
acdlite Oct 28, 2022
5450dd4
Strict Mode: Reuse memoized result from first pass (#25583)
acdlite Oct 28, 2022
d2c0ab1
In work loop, add enum of reasons for suspending
acdlite Oct 24, 2022
952dfff
Split suspended work loop logic into separate functions
acdlite Oct 24, 2022
bdd3d08
Extract logic for detecting bad fallback to helper
acdlite Oct 24, 2022
2ac77aa
Clean up vestige of useOpaqueIdentifier (#25587)
acdlite Oct 30, 2022
765805b
Fix type check for null (#25595)
sebmarkbage Oct 31, 2022
ab075a2
Do not unmount layout effects on initial Offscreen mount (#25592)
sammy-SC Nov 1, 2022
6883d79
[ServerRenderer] Setup for adding data attributes streaming format (#…
mofeiZ Nov 1, 2022
36426e6
Allow uncached IO to stablize (#25561)
acdlite Nov 1, 2022
5f7ef8c
[Float] handle resource Resource creation inside svg context (#25599)
gnoff Nov 1, 2022
8e69bc4
Make host context use null as empty and only error in dev (#25609)
sebmarkbage Nov 1, 2022
4ea063b
refactor isHostResourceType to not receive the context from reconcile…
gnoff Nov 2, 2022
1a90262
Unwrap sync resolved thenables without suspending (#25615)
acdlite Nov 2, 2022
1a08f14
[ServerRenderer] Move fizz external runtime implementation to react-d…
mofeiZ Nov 3, 2022
df61e70
Remove check in renderDidSuspendDelayIfPossible (#25630)
acdlite Nov 3, 2022
4bd245e
Do not unmount layout effects if ancestor Offscreen is hidden (#25628)
sammy-SC Nov 4, 2022
18dff79
[DevTools] add support for HostSingleton & HostResource (#25616)
mondaychen Nov 7, 2022
1e3e30d
Fix useSyncExternalStore dropped update when state is dispatched in r…
pandaiolo Nov 8, 2022
d1e35c7
Don't disappear layout effects unnecessarily (#25660)
sammy-SC Nov 10, 2022
c54e354
[DevTools] bug fix for Hydrating fibers (#25663)
mondaychen Nov 11, 2022
e1dd0a2
Remove recoverable error when a sync update flows into a dehydrated b…
sebmarkbage Nov 16, 2022
c343f80
[react-float] feature detect getRootNode (#25689)
kassens Nov 16, 2022
355dd7d
Bump loader-utils from 2.0.0 to 2.0.4 in /fixtures/flight (#25694)
dependabot[bot] Nov 16, 2022
db8a3fc
Bump loader-utils from 1.4.0 to 1.4.2 in /fixtures/fizz (#25680)
dependabot[bot] Nov 16, 2022
d65b88d
Eagerly initialize an mutable object for instance.refs (#25696)
sebmarkbage Nov 16, 2022
07f46ec
Turn on key spread warning in jsx-runtime for everyone (#25697)
sebmarkbage Nov 16, 2022
6fb8133
Turn on string ref deprecation warning for everybody (not codemoddabl…
eps1lon Nov 17, 2022
7b17f7b
Enable warning for defaultProps on function components for everyone (…
sebmarkbage Nov 17, 2022
44c4e6f
Force unwind work loop during selective hydration (#25695)
acdlite Nov 17, 2022
9dfbd9f
use: Don't suspend if there are pending updates
acdlite Nov 3, 2022
4a2d86b
Don't reset work loop until stack is unwound
acdlite Nov 2, 2022
5eb78d0
Pass ThenableState to replaySuspendedUnitOfWork
acdlite Nov 2, 2022
4387d75
Allow more hooks to be added when replaying mount
acdlite Nov 3, 2022
33e3d28
Reuse hooks when replaying a suspended component
acdlite Nov 3, 2022
6b4c031
Check thenable instead of thenableState
acdlite Nov 3, 2022
f284d9f
Track ThenableState alongside other hooks
acdlite Nov 3, 2022
f31005d
Add SyncHydrationLane (#25698)
tyao1 Nov 17, 2022
b0a6f66
Test for double invoke effect of useLayoutEffect
kassens Nov 17, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
[Float] handle noscript context for Resources (facebook#25559)
stacked on facebook#25569

On the client noscript already never renders children so no resources
will be extracted from this context. On the server we now track if we
are in a noscript context and turn off Resource semantics in this scope
  • Loading branch information
gnoff authored Oct 27, 2022
commit 09def5990b01d04433f852c9e6bc41039be052d9
4 changes: 4 additions & 0 deletions packages/react-dom-bindings/src/server/ReactDOMFloatServer.js
Original file line number Diff line number Diff line change
Expand Up @@ -863,6 +863,10 @@ export function resourcesFromLink(props: Props): boolean {
}
}
if (props.onLoad || props.onError) {
// When a link has these props we can't treat it is a Resource but if we rendered it on the
// server it would look like a Resource in the rendered html (the onLoad/onError aren't emitted)
// Instead we expect the client to insert them rather than hydrate them which also guarantees
// that the onLoad and onError won't fire before the event handlers are attached
return true;
}

Expand Down
128 changes: 108 additions & 20 deletions packages/react-dom-bindings/src/server/ReactDOMServerFormatConfig.js
Original file line number Diff line number Diff line change
Expand Up @@ -274,15 +274,18 @@ type InsertionMode = 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7;
export type FormatContext = {
insertionMode: InsertionMode, // root/svg/html/mathml/table
selectedValue: null | string | Array<string>, // the selected value(s) inside a <select>, or null outside <select>
noscriptTagInScope: boolean,
};

function createFormatContext(
insertionMode: InsertionMode,
selectedValue: null | string,
noscriptTagInScope: boolean,
): FormatContext {
return {
insertionMode,
selectedValue,
noscriptTagInScope,
};
}

Expand All @@ -293,7 +296,7 @@ export function createRootFormatContext(namespaceURI?: string): FormatContext {
: namespaceURI === 'http://www.w3.org/1998/Math/MathML'
? MATHML_MODE
: ROOT_HTML_MODE;
return createFormatContext(insertionMode, null);
return createFormatContext(insertionMode, null, false);
}

export function getChildFormatContext(
Expand All @@ -302,38 +305,77 @@ export function getChildFormatContext(
props: Object,
): FormatContext {
switch (type) {
case 'noscript':
return createFormatContext(HTML_MODE, null, true);
case 'select':
return createFormatContext(
HTML_MODE,
props.value != null ? props.value : props.defaultValue,
parentContext.noscriptTagInScope,
);
case 'svg':
return createFormatContext(SVG_MODE, null);
return createFormatContext(
SVG_MODE,
null,
parentContext.noscriptTagInScope,
);
case 'math':
return createFormatContext(MATHML_MODE, null);
return createFormatContext(
MATHML_MODE,
null,
parentContext.noscriptTagInScope,
);
case 'foreignObject':
return createFormatContext(HTML_MODE, null);
return createFormatContext(
HTML_MODE,
null,
parentContext.noscriptTagInScope,
);
// Table parents are special in that their children can only be created at all if they're
// wrapped in a table parent. So we need to encode that we're entering this mode.
case 'table':
return createFormatContext(HTML_TABLE_MODE, null);
return createFormatContext(
HTML_TABLE_MODE,
null,
parentContext.noscriptTagInScope,
);
case 'thead':
case 'tbody':
case 'tfoot':
return createFormatContext(HTML_TABLE_BODY_MODE, null);
return createFormatContext(
HTML_TABLE_BODY_MODE,
null,
parentContext.noscriptTagInScope,
);
case 'colgroup':
return createFormatContext(HTML_COLGROUP_MODE, null);
return createFormatContext(
HTML_COLGROUP_MODE,
null,
parentContext.noscriptTagInScope,
);
case 'tr':
return createFormatContext(HTML_TABLE_ROW_MODE, null);
return createFormatContext(
HTML_TABLE_ROW_MODE,
null,
parentContext.noscriptTagInScope,
);
}
if (parentContext.insertionMode >= HTML_TABLE_MODE) {
// Whatever tag this was, it wasn't a table parent or other special parent, so we must have
// entered plain HTML again.
return createFormatContext(HTML_MODE, null);
return createFormatContext(
HTML_MODE,
null,
parentContext.noscriptTagInScope,
);
}
if (parentContext.insertionMode === ROOT_HTML_MODE) {
// We've emitted the root and is now in plain HTML mode.
return createFormatContext(HTML_MODE, null);
return createFormatContext(
HTML_MODE,
null,
parentContext.noscriptTagInScope,
);
}
return parentContext;
}
Expand Down Expand Up @@ -1155,8 +1197,13 @@ function pushBase(
props: Object,
responseState: ResponseState,
textEmbedded: boolean,
noscriptTagInScope: boolean,
): ReactNodeList {
if (enableFloat && resourcesFromElement('base', props)) {
if (
enableFloat &&
!noscriptTagInScope &&
resourcesFromElement('base', props)
) {
if (textEmbedded) {
// This link follows text but we aren't writing a tag. while not as efficient as possible we need
// to be safe and assume text will follow by inserting a textSeparator
Expand All @@ -1175,8 +1222,13 @@ function pushMeta(
props: Object,
responseState: ResponseState,
textEmbedded: boolean,
noscriptTagInScope: boolean,
): ReactNodeList {
if (enableFloat && resourcesFromElement('meta', props)) {
if (
enableFloat &&
!noscriptTagInScope &&
resourcesFromElement('meta', props)
) {
if (textEmbedded) {
// This link follows text but we aren't writing a tag. while not as efficient as possible we need
// to be safe and assume text will follow by inserting a textSeparator
Expand All @@ -1195,8 +1247,9 @@ function pushLink(
props: Object,
responseState: ResponseState,
textEmbedded: boolean,
noscriptTagInScope: boolean,
): ReactNodeList {
if (enableFloat && resourcesFromLink(props)) {
if (enableFloat && !noscriptTagInScope && resourcesFromLink(props)) {
if (textEmbedded) {
// This link follows text but we aren't writing a tag. while not as efficient as possible we need
// to be safe and assume text will follow by inserting a textSeparator
Expand Down Expand Up @@ -1318,6 +1371,7 @@ function pushTitle(
target: Array<Chunk | PrecomputedChunk>,
props: Object,
responseState: ResponseState,
noscriptTagInScope: boolean,
): ReactNodeList {
if (__DEV__) {
const children = props.children;
Expand Down Expand Up @@ -1359,7 +1413,11 @@ function pushTitle(
}
}

if (enableFloat && resourcesFromElement('title', props)) {
if (
enableFloat &&
!noscriptTagInScope &&
resourcesFromElement('title', props)
) {
// We have converted this link exclusively to a resource and no longer
// need to emit it
return null;
Expand Down Expand Up @@ -1520,8 +1578,9 @@ function pushScript(
props: Object,
responseState: ResponseState,
textEmbedded: boolean,
noscriptTagInScope: boolean,
): null {
if (enableFloat && resourcesFromScript(props)) {
if (enableFloat && !noscriptTagInScope && resourcesFromScript(props)) {
if (textEmbedded) {
// This link follows text but we aren't writing a tag. while not as efficient as possible we need
// to be safe and assume text will follow by inserting a textSeparator
Expand Down Expand Up @@ -1863,18 +1922,47 @@ export function pushStartInstance(
return pushStartMenuItem(target, props, responseState);
case 'title':
return enableFloat
? pushTitle(target, props, responseState)
? pushTitle(
target,
props,
responseState,
formatContext.noscriptTagInScope,
)
: pushStartTitle(target, props, responseState);
case 'link':
return pushLink(target, props, responseState, textEmbedded);
return pushLink(
target,
props,
responseState,
textEmbedded,
formatContext.noscriptTagInScope,
);
case 'script':
return enableFloat
? pushScript(target, props, responseState, textEmbedded)
? pushScript(
target,
props,
responseState,
textEmbedded,
formatContext.noscriptTagInScope,
)
: pushStartGenericElement(target, props, type, responseState);
case 'meta':
return pushMeta(target, props, responseState, textEmbedded);
return pushMeta(
target,
props,
responseState,
textEmbedded,
formatContext.noscriptTagInScope,
);
case 'base':
return pushBase(target, props, responseState, textEmbedded);
return pushBase(
target,
props,
responseState,
textEmbedded,
formatContext.noscriptTagInScope,
);
// Newline eating tags
case 'listing':
case 'pre': {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,7 @@ export function createRootFormatContext(): FormatContext {
return {
insertionMode: HTML_MODE, // We skip the root mode because we don't want to emit the DOCTYPE in legacy mode.
selectedValue: null,
noscriptTagInScope: false,
};
}

Expand Down
Loading