From 2872a26e14627f90e812048f21feb6d0c8cc0244 Mon Sep 17 00:00:00 2001 From: Josh Story Date: Tue, 4 Oct 2022 16:11:15 -0700 Subject: [PATCH] track resources in different roots separately (#25388) * track resources in different roots separately * flow types * add test demonstrating portals deep into shadowRoots * revert hostcontext changes * lints * funge style cache key a la ReactDOMComponentTree * hide hacks in componentTree --- .../src/client/ReactDOMComponentTree.js | 10 ++ .../src/client/ReactDOMFloatClient.js | 99 +++++++++------ .../src/client/ReactDOMHostConfig.js | 3 +- .../src/__tests__/ReactDOMFloat-test.js | 116 ++++++++++++++++++ .../src/ReactFiberHostContext.js | 21 ++++ .../src/ReactFiberHostContext.new.js | 6 + .../src/ReactFiberHostContext.old.js | 6 + scripts/error-codes/codes.json | 2 +- 8 files changed, 220 insertions(+), 43 deletions(-) create mode 100644 packages/react-reconciler/src/ReactFiberHostContext.js diff --git a/packages/react-dom-bindings/src/client/ReactDOMComponentTree.js b/packages/react-dom-bindings/src/client/ReactDOMComponentTree.js index 064646a7c1f91..5c45949f97419 100644 --- a/packages/react-dom-bindings/src/client/ReactDOMComponentTree.js +++ b/packages/react-dom-bindings/src/client/ReactDOMComponentTree.js @@ -7,6 +7,7 @@ * @flow */ +import type {FloatRoot, StyleResource} from './ReactDOMFloatClient'; import type {Fiber} from 'react-reconciler/src/ReactInternalTypes'; import type {ReactScopeInstance} from 'shared/ReactTypes'; import type { @@ -42,6 +43,7 @@ const internalContainerInstanceKey = '__reactContainer$' + randomKey; const internalEventHandlersKey = '__reactEvents$' + randomKey; const internalEventHandlerListenersKey = '__reactListeners$' + randomKey; const internalEventHandlesSetKey = '__reactHandles$' + randomKey; +const internalRootNodeStylesSetKey = '__reactStyles$' + randomKey; export function detachDeletedInstance(node: Instance): void { // TODO: This function is only called on host components. I don't think all of @@ -266,3 +268,11 @@ export function doesTargetHaveEventHandle( } return eventHandles.has(eventHandle); } + +export function getStylesFromRoot(root: FloatRoot): Map { + let styles = (root: any)[internalRootNodeStylesSetKey]; + if (!styles) { + styles = (root: any)[internalRootNodeStylesSetKey] = new Map(); + } + return styles; +} diff --git a/packages/react-dom-bindings/src/client/ReactDOMFloatClient.js b/packages/react-dom-bindings/src/client/ReactDOMFloatClient.js index 500cdfb79f9c8..cf803261ab92f 100644 --- a/packages/react-dom-bindings/src/client/ReactDOMFloatClient.js +++ b/packages/react-dom-bindings/src/client/ReactDOMFloatClient.js @@ -7,9 +7,11 @@ * @flow */ -import type {Instance} from './ReactDOMHostConfig'; +import type {Instance, Container} from './ReactDOMHostConfig'; + import ReactDOMSharedInternals from 'shared/ReactDOMSharedInternals.js'; const {Dispatcher} = ReactDOMSharedInternals; +import {DOCUMENT_NODE} from '../shared/HTMLNodeType'; import { validateUnmatchedLinkResourceProps, validatePreloadResourceDifference, @@ -21,7 +23,9 @@ import { validatePreinitArguments, } from '../shared/ReactDOMResourceValidation'; import {createElement, setInitialProperties} from './ReactDOMComponent'; +import {getStylesFromRoot} from './ReactDOMComponentTree'; import {HTML_NAMESPACE} from '../shared/DOMNamespaces'; +import {getCurrentRootHostContainer} from 'react-reconciler/src/ReactFiberHostContext'; // The resource types we support. currently they match the form for the as argument. // In the future this may need to change, especially when modules / scripts are supported @@ -47,7 +51,7 @@ type StyleProps = { 'data-rprec': string, [string]: mixed, }; -type StyleResource = { +export type StyleResource = { type: 'style', // Ref count for resource @@ -66,7 +70,7 @@ type StyleResource = { loaded: boolean, error: mixed, instance: ?Element, - ownerDocument: Document, + root: FloatRoot, }; type Props = {[string]: mixed}; @@ -79,11 +83,6 @@ type Resource = StyleResource | PreloadResource; // e = errored type StyleResourceLoadingState = Promise & {s?: 'l' | 'e'}; -// When rendering we set the currentDocument if one exists. we use this for Resources -// we encounter during render. If this is null and we are dispatching preloads and -// other calls on the ReactDOM module we look for the window global and get the document from there -let currentDocument: ?Document = null; - // It is valid to preload even when we aren't actively rendering. For cases where Float functions are // called when there is no rendering we track the last used document. It is not safe to insert // arbitrary resources into the lastCurrentDocument b/c it may not actually be the document @@ -93,14 +92,17 @@ let currentDocument: ?Document = null; let lastCurrentDocument: ?Document = null; let previousDispatcher = null; -export function prepareToRenderResources(ownerDocument: Document) { - currentDocument = lastCurrentDocument = ownerDocument; +export function prepareToRenderResources(rootContainer: Container) { + // Flot thinks that getRootNode returns a Node but it actually returns a + // Document or ShadowRoot + const rootNode: FloatRoot = (rootContainer.getRootNode(): any); + lastCurrentDocument = getDocumentFromRoot(rootNode); + previousDispatcher = Dispatcher.current; Dispatcher.current = ReactDOMClientDispatcher; } export function cleanupAfterRenderResources() { - currentDocument = null; Dispatcher.current = previousDispatcher; previousDispatcher = null; } @@ -110,9 +112,16 @@ export function cleanupAfterRenderResources() { // from Internals -> ReactDOM -> FloatClient -> Internals so this doesn't introduce a new one. export const ReactDOMClientDispatcher = {preload, preinit}; +export type FloatRoot = Document | ShadowRoot; + // global maps of Resources const preloadResources: Map = new Map(); -const styleResources: Map = new Map(); + +function getCurrentResourceRoot(): null | FloatRoot { + const currentContainer = getCurrentRootHostContainer(); + // $FlowFixMe flow should know currentContainer is a Node and has getRootNode + return currentContainer ? currentContainer.getRootNode() : null; +} // Preloads are somewhat special. Even if we don't have the Document // used by the root that is rendering a component trying to insert a preload @@ -121,13 +130,22 @@ const styleResources: Map = new Map(); // lastCurrentDocument if that exists. As a fallback we will use the window.document // if available. function getDocumentForPreloads(): ?Document { - try { - return currentDocument || lastCurrentDocument || window.document; - } catch (error) { - return null; + const root = getCurrentResourceRoot(); + if (root) { + return root.ownerDocument || root; + } else { + try { + return lastCurrentDocument || window.document; + } catch (error) { + return null; + } } } +function getDocumentFromRoot(root: FloatRoot): Document { + return root.ownerDocument || root; +} + // -------------------------------------- // ReactDOM.Preload // -------------------------------------- @@ -200,8 +218,9 @@ function preinit(href: string, options: PreinitOptions) { typeof options === 'object' && options !== null ) { + const resourceRoot = getCurrentResourceRoot(); const as = options.as; - if (!currentDocument) { + if (!resourceRoot) { // We are going to emit a preload as a best effort fallback since this preinit // was called outside of a render. Given the passive nature of this fallback // we do not warn in dev when props disagree if there happens to already be a @@ -223,6 +242,7 @@ function preinit(href: string, options: PreinitOptions) { switch (as) { case 'style': { + const styleResources = getStylesFromRoot(resourceRoot); const precedence = options.precedence || 'default'; let resource = styleResources.get(href); if (resource) { @@ -241,8 +261,8 @@ function preinit(href: string, options: PreinitOptions) { options, ); resource = createStyleResource( - // $FlowFixMe[incompatible-call] found when upgrading Flow - currentDocument, + styleResources, + resourceRoot, href, precedence, resourceProps, @@ -303,9 +323,10 @@ export function getResource( pendingProps: Props, currentProps: null | Props, ): null | Resource { - if (!currentDocument) { + const resourceRoot = getCurrentResourceRoot(); + if (!resourceRoot) { throw new Error( - '"currentDocument" was expected to exist. This is a bug in React.', + '"resourceRoot" was expected to exist. This is a bug in React.', ); } switch (type) { @@ -313,6 +334,7 @@ export function getResource( const {rel} = pendingProps; switch (rel) { case 'stylesheet': { + const styleResources = getStylesFromRoot(resourceRoot); let didWarn; if (__DEV__) { if (currentProps) { @@ -348,8 +370,8 @@ export function getResource( } else { const resourceProps = stylePropsFromRawProps(styleRawProps); resource = createStyleResource( - // $FlowFixMe[incompatible-call] found when upgrading Flow - currentDocument, + styleResources, + resourceRoot, href, precedence, resourceProps, @@ -384,8 +406,7 @@ export function getResource( } else { const resourceProps = preloadPropsFromRawProps(preloadRawProps); resource = createPreloadResource( - // $FlowFixMe[incompatible-call] found when upgrading Flow - currentDocument, + getDocumentFromRoot(resourceRoot), href, resourceProps, ); @@ -463,7 +484,8 @@ function createResourceInstance( } function createStyleResource( - ownerDocument: Document, + styleResources: Map, + root: FloatRoot, href: string, precedence: string, props: StyleProps, @@ -479,7 +501,7 @@ function createStyleResource( const limitedEscapedHref = escapeSelectorAttributeValueInsideDoubleQuotes( href, ); - const existingEl = ownerDocument.querySelector( + const existingEl = root.querySelector( `link[rel="stylesheet"][href="${limitedEscapedHref}"]`, ); const resource = { @@ -492,7 +514,7 @@ function createStyleResource( preloaded: false, loaded: false, error: false, - ownerDocument, + root, instance: null, }; styleResources.set(href, resource); @@ -567,7 +589,7 @@ function immediatelyPreloadStyleResource(resource: StyleResource) { const {href, props} = resource; const preloadProps = preloadPropsFromStyleProps(props); resource.hint = createPreloadResource( - resource.ownerDocument, + getDocumentFromRoot(resource.root), href, preloadProps, ); @@ -613,11 +635,11 @@ function createPreloadResource( function acquireStyleResource(resource: StyleResource): Instance { if (!resource.instance) { - const {props, ownerDocument, precedence} = resource; + const {props, root, precedence} = resource; const limitedEscapedHref = escapeSelectorAttributeValueInsideDoubleQuotes( props.href, ); - const existingEl = ownerDocument.querySelector( + const existingEl = root.querySelector( `link[rel="stylesheet"][data-rprec][href="${limitedEscapedHref}"]`, ); if (existingEl) { @@ -649,11 +671,11 @@ function acquireStyleResource(resource: StyleResource): Instance { const instance = createResourceInstance( 'link', resource.props, - ownerDocument, + getDocumentFromRoot(root), ); attachLoadListeners(instance, resource); - insertStyleInstance(instance, precedence, ownerDocument); + insertStyleInstance(instance, precedence, root); resource.instance = instance; } } @@ -724,11 +746,9 @@ function onResourceError( function insertStyleInstance( instance: Instance, precedence: string, - ownerDocument: Document, + root: FloatRoot, ): void { - const nodes = ownerDocument.querySelectorAll( - 'link[rel="stylesheet"][data-rprec]', - ); + const nodes = root.querySelectorAll('link[rel="stylesheet"][data-rprec]'); const last = nodes.length ? nodes[nodes.length - 1] : null; let prior = last; for (let i = 0; i < nodes.length; i++) { @@ -746,9 +766,8 @@ function insertStyleInstance( // must exist. ((prior.parentNode: any): Node).insertBefore(instance, prior.nextSibling); } else { - // @TODO call getRootNode on root.container. if it is a Document, insert into head - // if it is a ShadowRoot insert it into the root node. - const parent = ownerDocument.head; + const parent = + root.nodeType === DOCUMENT_NODE ? ((root: any): Document).head : root; if (parent) { parent.insertBefore(instance, parent.firstChild); } else { diff --git a/packages/react-dom-bindings/src/client/ReactDOMHostConfig.js b/packages/react-dom-bindings/src/client/ReactDOMHostConfig.js index 63c854a92acdd..780fb59e571d5 100644 --- a/packages/react-dom-bindings/src/client/ReactDOMHostConfig.js +++ b/packages/react-dom-bindings/src/client/ReactDOMHostConfig.js @@ -41,7 +41,6 @@ import { warnForDeletedHydratableText, warnForInsertedHydratedElement, warnForInsertedHydratedText, - getOwnerDocumentFromRootContainer, } from './ReactDOMComponent'; import {getSelectionInformation, restoreSelection} from './ReactInputSelection'; import setTextContent from './setTextContent'; @@ -1376,7 +1375,7 @@ function isHostResourceInstance(instance: Instance | Container): boolean { export function prepareRendererToRender(rootContainer: Container) { if (enableFloat) { - prepareToRenderResources(getOwnerDocumentFromRootContainer(rootContainer)); + prepareToRenderResources(rootContainer); } } diff --git a/packages/react-dom/src/__tests__/ReactDOMFloat-test.js b/packages/react-dom/src/__tests__/ReactDOMFloat-test.js index 3484c172c4cf2..6c45c68839a14 100644 --- a/packages/react-dom/src/__tests__/ReactDOMFloat-test.js +++ b/packages/react-dom/src/__tests__/ReactDOMFloat-test.js @@ -552,6 +552,122 @@ describe('ReactDOMFloat', () => { }); }); + describe('document encapsulation', () => { + // @gate enableFloat + it('can support styles inside portals to a shadowRoot', async () => { + const shadow = document.body.attachShadow({mode: 'open'}); + const root = ReactDOMClient.createRoot(container); + root.render( + <> + + {ReactDOM.createPortal( +
+ + shadow +
, + shadow, + )} + container + , + ); + expect(Scheduler).toFlushWithoutYielding(); + expect(getVisibleChildren(document)).toEqual( + + + + + + +
container
+ + , + ); + expect(getVisibleChildren(shadow)).toEqual([ + , +
shadow
, + ]); + }); + // @gate enableFloat + it('can support styles inside portals to an element in shadowRoots', async () => { + const template = document.createElement('template'); + template.innerHTML = + "
"; + const shadow = document.body.attachShadow({mode: 'open'}); + shadow.appendChild(template.content); + + const shadowContainer1 = shadow.getElementById('shadowcontainer1'); + const shadowContainer2 = shadow.getElementById('shadowcontainer2'); + const root = ReactDOMClient.createRoot(container); + root.render( + <> + + {ReactDOM.createPortal( +
+ + 1 +
, + shadow, + )} + {ReactDOM.createPortal( +
+ + 2 +
, + shadowContainer1, + )} + {ReactDOM.createPortal( +
+ + 3 +
, + shadowContainer2, + )} + container + , + ); + expect(Scheduler).toFlushWithoutYielding(); + expect(getVisibleChildren(document)).toEqual( + + + + + + + + + +
container
+ + , + ); + expect(getVisibleChildren(shadow)).toEqual([ + , + , + , + , +
+
+
2
+
+
+
3
+
+
, +
1
, + ]); + }); + }); + describe('style resources', () => { // @gate enableFloat it('treats link rel stylesheet elements as a style resource when it includes a precedence when server rendering', async () => { diff --git a/packages/react-reconciler/src/ReactFiberHostContext.js b/packages/react-reconciler/src/ReactFiberHostContext.js new file mode 100644 index 0000000000000..461ecdd492ab3 --- /dev/null +++ b/packages/react-reconciler/src/ReactFiberHostContext.js @@ -0,0 +1,21 @@ +/** + * Copyright (c) Facebook, Inc. and its affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + * + * @flow + */ + +import type {Container} from './ReactFiberHostConfig'; +import {enableNewReconciler} from 'shared/ReactFeatureFlags'; + +import {getCurrentRootHostContainer as getCurrentRootHostContainer_old} from './ReactFiberHostContext.old'; + +import {getCurrentRootHostContainer as getCurrentRootHostContainer_new} from './ReactFiberHostContext.new'; + +export function getCurrentRootHostContainer(): null | Container { + return enableNewReconciler + ? getCurrentRootHostContainer_new() + : getCurrentRootHostContainer_old(); +} diff --git a/packages/react-reconciler/src/ReactFiberHostContext.new.js b/packages/react-reconciler/src/ReactFiberHostContext.new.js index 282414a658cba..1ea596222bfa5 100644 --- a/packages/react-reconciler/src/ReactFiberHostContext.new.js +++ b/packages/react-reconciler/src/ReactFiberHostContext.new.js @@ -38,6 +38,11 @@ function requiredContext(c: Value | NoContextT): Value { return (c: any); } +function getCurrentRootHostContainer(): null | Container { + const container = rootInstanceStackCursor.current; + return container === NO_CONTEXT ? null : (container: any); +} + function getRootHostContainer(): Container { const rootInstance = requiredContext(rootInstanceStackCursor.current); return rootInstance; @@ -101,6 +106,7 @@ function popHostContext(fiber: Fiber): void { } export { + getCurrentRootHostContainer, getHostContext, getRootHostContainer, popHostContainer, diff --git a/packages/react-reconciler/src/ReactFiberHostContext.old.js b/packages/react-reconciler/src/ReactFiberHostContext.old.js index a4735b4bb6976..293c432e032fc 100644 --- a/packages/react-reconciler/src/ReactFiberHostContext.old.js +++ b/packages/react-reconciler/src/ReactFiberHostContext.old.js @@ -38,6 +38,11 @@ function requiredContext(c: Value | NoContextT): Value { return (c: any); } +function getCurrentRootHostContainer(): null | Container { + const container = rootInstanceStackCursor.current; + return container === NO_CONTEXT ? null : (container: any); +} + function getRootHostContainer(): Container { const rootInstance = requiredContext(rootInstanceStackCursor.current); return rootInstance; @@ -101,6 +106,7 @@ function popHostContext(fiber: Fiber): void { } export { + getCurrentRootHostContainer, getHostContext, getRootHostContainer, popHostContainer, diff --git a/scripts/error-codes/codes.json b/scripts/error-codes/codes.json index 1112e284460e0..48fcf61717273 100644 --- a/scripts/error-codes/codes.json +++ b/scripts/error-codes/codes.json @@ -431,6 +431,6 @@ "443": "acquireResource encountered a resource type it did not expect: \"%s\". this is a bug in React.", "444": "getResource encountered a resource type it did not expect: \"%s\". this is a bug in React.", "445": "\"currentResources\" was expected to exist. This is a bug in React.", - "446": "\"currentDocument\" was expected to exist. This is a bug in React.", + "446": "\"resourceRoot\" was expected to exist. This is a bug in React.", "447": "While attempting to insert a Resource, React expected the Document to contain a head element but it was not found." }